51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 73600|回复: 157
打印 上一主题 下一主题

QTP识别和操作对象的原理(http://www.yabest.net)

[复制链接]

该用户从未签到

跳转到指定楼层
1#
发表于 2007-8-16 16:11:02 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
呵呵,这个文章本来是在给别人的帖《GetROProperty,GetTOProperties,GetTOProperty的区别 》回复时写的,没有独立出来。
http://bbs.51testing.com/viewthread.php?tid=13554&page=1#pid369327
写完后,在网上被多处转载。没想到今天又被转贴回51testing来了。

虽然在精华版里,也赚了几朵花,但没独立总是不方便。所以修整了一下,独立成帖,以方便大家阅读。
欢迎大家拍砖!(当然更欢迎大家送花sdlkfj5)


一、QTP识别对象的原理(by yabest, http://www.yabest.net

QTP里的对象有两个概念,一个是Test Object(简称TO),一个是Runtime Object(简称RO)。
这两个概念从字面上不大好理解,也容易混淆。
但从实际作用上来看,应该说TO就是是仓库文件里定义的仓库对象,RO是被测试软件的实际对象。

QTP识别对象,一般是要求先在对象仓库文件里定义仓库对象,里面存有实际对象的特征属性的值。
然后在运行的时候,QTP会根据脚本里的对象名字,在对象仓库里找到对应的仓库对象,
接着根据仓库对象的特征属性描述,在被测试软件里搜索找到相匹配的实际对象,最后就可以对实际对象进行操作了。

仓库对象TO一般在录制/编写脚本时加入仓库文件,它不仅可以在录制编写时进行修改,
也可以在运行过程中进行动态修改,以匹配实际对象。

和TO、RO相关的几个函数有:

GetTOProperty():取得仓库对象的某个属性的值
GetTOProperties():取得仓库对象的所有属性的值
SetTOProperty():设置仓库对象的某个属性的值

GetROProperty():取得实际对象的某个属性的值

(注:这几个函数访问的都是对象的封装属性,不是对象的自身属性,封装属性和自身属性的区别详见后面第二章QTP操作对象的原理)

理解了TO的含义,你就可以自由的用SetTOProperty()定义TO,以灵活的操作RO

比如有个测试任务,窗口上有很多待检查的记录,每条记录右边都有一个Check按钮,用来检查各条记录。
记录个数不定,所以Check按钮个数也就不定,只有一个Edit显示记录个数。
我们要对每条记录进行检查,也就是要点击每个Check按钮。
但是Check按钮个数不定,不好录制,而且个数可能也很多(上百个),即使能一一录制,那也很麻烦。

那我有一个好办法,只录制一个按钮对象,它设有两个特征属性 label=OK, index=0
然后用下面的脚本,就可以完成测试

buttonNum = CInt(JavaWindow("Test").JavaEdit("Record Num").GetROProperty("value"))
For buttonIndex = 0 to buttonNum - 1
  JavaWindow("Test").JavaButton("Check").SetTOProperty("index", buttonIndex)
  JavaWindow("Test").JavaButton("Check").Click
Next


或者窗口上有New、Modify、Delete、Check等好几个按钮,要把这几个按钮一一按过去
我在对象仓库里只设置一个按钮对象AnyButton,label特征属性值填任意值,然后用下面脚本执行测试

JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "New")
JavaWindow("Test").JavaButton("AnyButton").Click

JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "Modify")
JavaWindow("Test").JavaButton("AnyButton").Click

JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "Delete")
JavaWindow("Test").JavaButton("AnyButton").Click

JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "Check")
JavaWindow("Test").JavaButton("AnyButton").Click

另外,QTP还支持脚本描述的方法来定义和访问对象,即不需要在仓库里定义,也能访问和操作实际对象
( Written by yabest,http://www.yabest.net

如上面两个任务,可以如下实现

1. 不需要在仓库里定义Check按钮对象,直接用下面脚本来实现测试

buttonNum = CInt(JavaWindow("Test").JavaEdit("Record Num").GetROProperty("value"))
For buttonIndex = 0 to buttonNum - 1
  JavaWindow("Test").JavaButton("label:=Check", "index:="+CStr(buttonIndex)).Click
Next

2. 不需要在仓库里定义New、Modify、Delete、Check按钮对象,直接用下面脚本来实现测试

JavaWindow("Test").JavaButton("label:=New").Click
JavaWindow("Test").JavaButton("label:=Modify").Click
JavaWindow("Test").JavaButton("label:=Delete").Click
JavaWindow("Test").JavaButton("label:=Check").Click



二、QTP操作对象的原理(by yabest, http://www.yabest.net

QTP为用户提供了两种操作对象的接口,一种就是对象的封装接口,另一种是对象的自身接口。

对象的自身接口是对象控件本身的接口,只要做过软件开发,使用过控件的人应该很清楚。
对象的封装接口是QTP为对象封装的另一层接口,它是QTP通过调用对象的自身接口来实现的。


两种接口的脚本书写格式的差别在于:
  自身接口需要在对象名后面加object再加属性名或方法名,
  封装接口就不用在对象名后面加object。

具体格式如下:
  对实际对象的操作:
      对象.object.自身属性
      对象.object.自身方法()
      对象.GetROProperty("封装属性")
      对象.封装方法()

  对仓库对象的操作:
      对象.GetTOProperty("封装属性")
      对象.GetTOProperties()      ’获取所有封装属性的值
      对象.SetTOProperty("封装属性", "封装属性值")

比如操作JavaEdit对象,通过QTP封装的封装接口,脚本如下:
  设置JavaEdit的内容:
     JavaDialog("Add NE").JavaEdit("NE Name").Set "NE1"
  读取JavaEdit的内容:
     msgbox JavaDialog("Add NE").JavaEdit("NE Name").GetROProperty("value")

如果通过JavaEdit的自身接口,脚本如下:
  设置JavaEdit的内容:
     JavaDialog("Add NE").JavaEdit("NE Name").object.setText("NE1")
  读取JavaEdit的内容:
     Msgbox JavaDialog("Add NE").JavaEdit("NE Name").object.getText()

QTP执行JavaEdit().Set语句时,是通过执行JavaEdit().object.setText()来实现的。
QTP执行JavaEdit().GetROProperty("value"),是通过执行JavaEdit().object.getText()来实现的。
JavaEdit对象的封装接口Set()和GetROProperty("value"),是QTP封装JavaEdit对象的自身接口setText()和getText()而得来的。

对象的封装接口是QTP使用的缺省接口,我们录制出来的脚本都是使用封装接口,大家用的也都是封装接口。
但是封装接口不如自身接口丰富,因为QTP只是封装了部分常用的自身接口嘛。
所以我们在需要时,可以绕过封装接口,直接调用对象的自身接口。
不过有些自身接口不够稳定,在实践中偶尔会出现问题,但是概率很少。
封装接口有相应功能的话,就尽量用封装接口吧!
( Written by yabest,http://www.yabest.net

理解了封装接口和自身接口的原理,我们就可以更加灵活的操作对象了。

但是我们怎么知道对象都有哪些封装接口和自身接口呢?
其实很简单,用对象查看器(Object Spy)查看对象,在查看窗口里有列出这些接口,包括属性和方法。
窗口中间有选择栏让你选择Run-time Object或者Test Object,
当你选择Runtime Object时,它显示的就是对象的自身接口(自身的属性和方法)
当你选择Test Object时,它显示的就是对象的封装接口(封装的属性和方法)

(注意:GetROProperty访问的是实际对象的封装接口,GetTOProperty访问的是仓库对象的封装接口,
            两者访问的都是对象的封装接口,即Object Spy窗口里选Test Object时显示的属性。
            不要以为GetROProperty访问的是自身接口,即Object Spy窗口里选Run-time Object时显示的属性。
            QTP里的Test Object/Run-time Object的概念太容易让人混淆了!
            它既用来区分仓库对象和实际对象,又用来区分对象的封装接口和自身接口。



明白了这些,你还等什么呢?快拿起对象查看器,看看对象都有哪些封装接口和自身接口,肆意的操作它,玩弄它吧!

比如执行
  JavaDialog("Add NE").JavaEdit("NE Name").object.setVisible(false)
哈哈,你的JavaEdit对象就当场消失不见了!!!

你可以拿这个做恶作剧,指着这个窗口逼问开发人员,JavaEdit对象哪去了?
开发人员肯定瞪大眼睛看着这个窗口,当场翘掉!!!

[ 本帖最后由 yabest 于 2008-8-1 10:37 编辑 ]
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏
回复

使用道具 举报

该用户从未签到

2#
发表于 2007-8-16 16:24:14 | 只看该作者

回复 #1 yabest 的帖子

支持一下,讲得很细致
回复 支持 反对

使用道具 举报

该用户从未签到

3#
发表于 2007-8-16 17:15:08 | 只看该作者
讲的很好,怎么没有人回帖?
回复 支持 反对

使用道具 举报

该用户从未签到

4#
发表于 2007-8-16 17:37:43 | 只看该作者
讲的很好哦.
回复 支持 反对

使用道具 举报

该用户从未签到

5#
发表于 2007-8-16 17:42:37 | 只看该作者
学到很多!lz对QTP的研究很深啊,以前是不是程序员啊?
回复 支持 反对

使用道具 举报

该用户从未签到

6#
发表于 2007-8-16 17:53:53 | 只看该作者
请教:
如果直接在QTP的脚本中使用类似下面的语句,会提示找不到对象,是不是不能直接使用QTP脚本啊。
---------------------------------------
2. 不需要在仓库里定义New、Modify、Delete、Check按钮对象,直接用下面脚本来实现测试

JavaWindow("Test").JavaButton("label:=New").Click
JavaWindow("Test").JavaButton("label:=Modify").Click
JavaWindow("Test").JavaButton("label:=Delete").Click
JavaWindow("Test").JavaButton("label:=Check").Click
----------------------------------------

我的例子:(注:其中Browser、Page、Frame对象都是存在的,只是从仓库中删除了对象WebEdit。)
Browser("计费系统").Page("计费系统").Frame("mainFrame").WebEdit("name=names").Set "王平"
回复 支持 反对

使用道具 举报

该用户从未签到

7#
发表于 2007-8-16 17:57:14 | 只看该作者
是不是先要通过 Description.Create() 来创建对象?
回复 支持 反对

使用道具 举报

该用户从未签到

8#
 楼主| 发表于 2007-8-16 18:00:56 | 只看该作者
原帖由 vickywang_no1 于 2007-8-16 17:53 发表
请教:
...
Browser("计费系统").Page("计费系统").Frame("mainFrame").WebEdit("name=names").Set "王平"
...


你是不是写错了,少了个冒号啊? 应该是 WebEdit("name:=names")
如果不行,可能有多个对象符合条件,那就加上index条件,如 WebEdit("name:=names", "index:=0“)
回复 支持 反对

使用道具 举报

该用户从未签到

9#
 楼主| 发表于 2007-8-16 20:22:53 | 只看该作者
原帖由 dionysus 于 2007-8-16 17:42 发表
学到很多!lz对QTP的研究很深啊,以前是不是程序员啊?


本来是做开发的,结果被骗来做了自动化 sdlkfj7
回复 支持 反对

使用道具 举报

该用户从未签到

10#
发表于 2007-8-16 21:48:01 | 只看该作者
怪不得,能看得出编程功底很强,对许多原理都了解的很清楚。开发的转来做自动化有先天的优势,也许会在这方面有更好的突破呢sdlkfj3
回复 支持 反对

使用道具 举报

  • TA的每日心情
    开心
    2016-2-27 08:48
  • 签到天数: 2 天

    连续签到: 1 天

    [LV.1]测试小兵

    11#
    发表于 2007-8-16 23:44:57 | 只看该作者
    谢谢 yabest  的共享! 这些资料对每一位学习和使用QTP的人都会很有帮助的
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    12#
    发表于 2007-8-17 09:38:38 | 只看该作者
    原帖由 yabest 于 2007-8-16 18:00 发表


    你是不是写错了,少了个冒号啊? 应该是 WebEdit("name:=names")
    如果不行,可能有多个对象符合条件,那就加上index条件,如 WebEdit("name:=names", "index:=0“)


    真是少了冒号。第1次知道QTP还有这样的功能,很神奇。太谢谢楼主了。
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    13#
    发表于 2007-8-17 09:48:33 | 只看该作者
    请教:
    --------------------------------------
    “另外,QTP还支持脚本描述的方法来定义和访问对象,即不需要在仓库里定义,也能访问和操作实际对象”
    ---------------------------------------
    这部分的QTP帮助在哪儿啊,我怎么没有找到呢?
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    14#
    发表于 2007-8-17 14:09:07 | 只看该作者
    已找到相关帮助:
    Mercury QuickTest Professional User's Guide > Working with Advanced Testing Features > Working with the Expert View and Function Library Windows > Using Programmatic Descriptions

    [ 本帖最后由 vickywang_no1 于 2007-9-3 11:45 编辑 ]

    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有帐号?(注-册)加入51Testing

    x
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    15#
     楼主| 发表于 2007-8-17 14:43:49 | 只看该作者
    原帖由 vickywang_no1 于 2007-8-17 09:38 发表

    第1次知道QTP还有这样的功能,很神奇。太谢谢楼主了。


    还是有很多人不知道描述性编程啊?!
    不过我是提倡多用对象库,少用描述性编程的哦!
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    16#
    发表于 2007-8-17 15:05:08 | 只看该作者
    不错,学习一下。
    最后的恶作剧很搞笑,有没人找个时间试试,呵呵sdlkfj3
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    17#
    发表于 2007-8-17 17:50:58 | 只看该作者
    强 ,狂dingsdlkfj3
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    18#
    发表于 2007-8-18 11:48:37 | 只看该作者
    不错,希望LZ多发这类帖子sdlkfj1
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    19#
    发表于 2007-8-18 11:50:59 | 只看该作者
    原帖由 yabest 于 2007-8-17 14:43 发表


    还是有很多人不知道描述性编程啊?!
    不过我是提倡多用对象库,少用描述性编程的哦!

    为什么呢?
    描述性编程我觉得挺不错啊sdlkfj5
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    20#
     楼主| 发表于 2007-8-18 14:15:09 | 只看该作者
    原帖由 Banditu 于 2007-8-18 11:50 发表

    为什么呢?
    描述性编程我觉得挺不错啊sdlkfj5


    唉,不说了!
    之前我是逢人就劝,结果就被人家说是偏激了sdlkfj7

    我也不想变成祥林嫂,唠叨个不停!
    你要是觉得好用,就用吧!

    [ 本帖最后由 yabest 于 2007-8-18 17:59 编辑 ]
    回复 支持 反对

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-11-8 07:55 , Processed in 0.085812 second(s), 27 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表