转载原文:https://www.jianshu.com/p/efe9dcf8bbaf
一、定位工具
在安装appium环境的时候我们已经安装了SDK,里面就自带有元素定位的工具,位置在.../sdk/tools/uiautomatorviewer.bat,由于安装路径各有不同,如果实在找不到可以具体路径,可以在电脑中查找uiautomatorviewer.bat文件,找到后双击打开即可,初始界面如下:
打开工具后,将手机连接电脑,这里要提醒的是,该工具只能获取手机当前的界面截图,而无法实时同步的显示手机界面,工具上有两个按钮:第一个图标是获取设备截屏,可以获取元素完整的层级关系。
第二个图标是获取压缩后的设备截屏,包含各个元素的属性,无法获得完整的层级路径
所以我们一般使用第一个获取设备截屏的按钮就行,连接好手机后点击第一个按钮得到当前手机界面,可以通过鼠标定位界面上的各个元素,右侧可查看该元素的相关值
ps:有一点要特别提醒,由于在自动化运行过程中一定要运行Appium,但我在使用中发现uiautomatorviewer在获取设备截图时好像与Appium有冲突,经常会报错:Error obtaining UI hierarchy。如果是在脚本调试阶段,可以停止运行Appium即可,调试完成后再开启Appium跑脚本
二、定位方法
1、ID定位(取resource-id、id、name)
当定位到元素后,在node detail中有该元素的所有参数,如果有resource-id属性就取resource-id值,有id属性就取id值,有name属性就取name值
但是在取resource-id、id、name值时这些值一定要唯一,可以唯一定位到该元素才行,如果值是一个通用值,那么无法区别出元素仍然无法定位,这时就要换其他定位方法
2、class、text定位(取class、text)
class是一个类属性,一般重复性会很高,所以很少直接使用,这里讲class属性是由于在xpath中会频繁用到class属性
text属性是文本,如果是要定位到唯一的标题之类的可以直接使用text属性
3、xpath定位
通过上面两种方法可知,如果有id属性当然就直接使用id值,但如果没有就干脆使用xpath,理论上xpath可以定位到所有元素
xpath的写法是根据层级的class属性一级一级定位到元素
绝对路径:
元素的全路径,包含了全部节点。这种方法首先一看就觉得太长写起来好累,如果层级再多那写起来简直无休无止了;另一方面由于涉及到太多层,一旦中间任何一层有变动,那元素就定位不到了,例如:
//android.widget.FrameLayout/android.widget.LinearLayout/android.widget.FrameLayout/android.widget.LinearLayout/android.widget.FrameLayout/android.widget.LinearLayout/android.view.ViewGroup/android.widget.ImageButton
相对路径:
1.如果当前class内存在唯一text可以定位元素,直接用当前class+text,例如://android.widget.TextView[@text='日常巡查']
2.如果当前class内存在唯一content-desc可以定位元素,直接用当前class+content-desc,例如://android.widget.TextView[@content-desc='巡场排查']
3.如果当前class内,resource-id、text、content-desc三者取其二或者取其三能唯一定位元素,直接用当前class+两者/三者并列,例如://android.widget.TextView[@resource-id='com.space.grid.debug:id/text' and @text='数据采集']
4.如果当前class内,text、content-desc中的文本内容不是完全符合,但能匹配部分内容,可用当前class+模糊定位contains,例如://android.widget.TextView[contains(@text,'重点管控')]
5.如果在当前class内无法通过以上4种办法定位元素,则向上寻找元素的父节点或者祖父节点通过以上4种办法定位,总之是找到能唯一定位且离元素最近的一个上层节点,例如://android.widget.RelativeLayout[@resource-id='com.space.grid.debug:id/address']/android.widget.ImageView[2]
ps:这里特别要强调的一点是android.widget.ImageView的index=1,但为什么写法却是android.widget.ImageView[2]下标为2,因为xpath的下标是从1开始的,没有0
4、坐标点定位
如果上面说的几种办法都无法定位,那只能采用终极办法,通过像素的坐标位置来定位元素了,比方我有个表单的选项是浮层,但工具没办法获取到浮层中每个项,这个时候只能通过关键字Click A Point [ x | y]来定位元素
PS:但有一点要注意的是,由于不同手机的分辨率不同,坐标位置也是会有一定的差异,同一个脚本在不同手机上运行可能会有不同的结果