一、QTP识别对象原理
QTP里的对象有两个概念,一个是Testtime Object(简称TO),一个是RunTime Object(简称RO)。
从实际作用来看,TO就是对象库中定义的对象,RO时被测试软件运行时的实际对象。
QTP识别对象,一般要求先在对象库中定义对象,里面存有实际对象的特征属性的值。然后在运行时,QTP会根据脚本里的对象名字,在对象库中找到对应的对象,接着根据对象库中的对象的特征属性描述,在被测试软件中搜索到相匹配的实际对象,最后就可以对实际对象进行操作了。
TO对象一般在录制/编写脚本时加入对象库,它不仅可以在录制编写时进行修改,也可以在运行过程中进行动态修改,以匹配实际对象。
和TO、RO相关的几个函数:
- SetToProperty:修改对象库中对象的属性值,但仅仅是临时修改,一旦本次action结束,对象库中该对象的属性恢复原值。
- GetToProperty:获得当前action的对象库中该对象的单一属性的值。
- GetToProperties:获得当前action的对象库中该对象的搜有属性的值。
- GetRoProperty:获得实时操作对象的属性值。
记录个数不定,所以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").SetTOProperty("label", "Modify") JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "Delete") JavaWindow("Test").JavaButton("AnyButton").SetTOProperty("label", "Check") |
另外,QTP还支持脚本描述的方法来定义和访问对象,即不需要在仓库里定义,也能访问和操作实际对象。
如上面两个任务,可以如下实现
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 |
二、QTP操作对象的原理
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只是封装了部分常用的自身接口嘛。
所以我们在需要时,可以绕过封装接口,直接调用对象的自身接口。
不过有些自身接口不够稳定,在实践中偶尔会出现问题,但是概率很少。
封装接口有相应功能的话,就尽量用封装接口吧!
理解了封装接口和自身接口的原理,我们就可以更加灵活的操作对象了。
但是我们怎么知道对象都有哪些封装接口和自身接口呢?
其实很简单,用对象查看器(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的概念太容易让人混淆了!
它既用来区分仓库对象和实际对象,又用来区分对象的封装接口和自身接口。)