xpath定位元素方法_App自动化之dom结构和元素定位方式(包含滑动列表定位)

c5a5f915976f40e4ab5139a47380a80a.gif

a078c57046012815540d730c69b2293f.png

本文为霍格沃兹测试学院学院学员课程学习笔记,系统学习文末加群。

1. DOM结构

59959262407a5418f7ed94fff6a5c6f4.png

先来看几个名词和解释:

  • dom: Document Object Model 文档对象模型

  • dom应用: 最早应用于htmljs的交互。界面的结构化描述, 常见的格式为htmlxml。核心元素为节点属性

  • xpath: xml路径语言,用于xml 中的节点定位,XPath 可在 xml 文档中对元素和属性进行遍历如下我们再来看一个Appdom:

0b55a439ff754f9768ce9b55af4ea1d1.png

控件的基础知识和selenium一样,appium为移动端抽象出了一个控件模型,称为dom结构;会把所有的控件都理解为xml文件,在xml文件里,每个控件都有自己的类型属性

既然有了类型和属性,自然就可以根据这些来定位元素,又因为整个模型是xml,也就同样可以通过xpath的方法来定位各个控件的信息了,是不是似曾相识?在Web端自动化时候也介绍过相关元素定位方式,具体可在文章末尾往期回顾第一条点击查看。

2. 控件定位

59959262407a5418f7ed94fff6a5c6f4.png

UI自动化测试的步骤三要素是: 

  • 定位

  • 交互 

  • 断言 

那么第一步便是要对元素进行定位,下面就来看看移动端如何进行元素定位

2.1 控件属性

通过uiautomatorviewer对雪球App首页的解析得到如下图结果:

b17f26e0b80f6f94a7cd1eee8ff4148b.png

通过解析结果我们可以看到元素的属性和类型有:

  • node

  • attribute

    • clickable

    • content-desc

    • resource-id

    • text

    • bounds

IOS和Android在控件属性和上稍微有些不同(这里先说个概括,后续单独出IOS的文章加以说明,欢迎关注):

  • dom属性和节点结构类似

  • 名字和属性的命名不同

2.2 定位方式

Appium 支持 WebDriver 定位策略的子集:

2.21 通过 “class” 查找 (例如, UI 组件的类型)-一般不推荐

这种就是通过判断控件类型来查找,例如TextView、ImageView等

6038aa67cf8c2fa228913460cff4db85.png 

在实际工作中,这种定位方式几乎不用,因为一个页面中可能会有很多的TextViewImageView等;

appiumdriver.findElementByClassName("android.widget.TextView");
2.22 通过 “xpath” 查找 (例如, 一个元素的路径以抽象的方式去表达,具有一定的约束)-重要

如上所述,xpath是不仅可以在移动端进行元素定位,并且是我们最常用的定位方式之一,在web端自动化我们会首推CSS定位,而在移动端定位我们会首推xpath定位,良好的xpath定位语法会给我们定位带来准确度和便利度,对速度的影响也完全会在我们的接受范围以内

如下dom结构中,一个界面上有多同类型控件,这些控件有相同的id属性,不具备唯一性,所以无法直接进行指定控件的定位操作,这个时候就该xpath大显身手了

64cf31d101edabf57140a88558248aa1.png 
上图可以看到,所有勾选控件的结构是一样的,相对位置是固定的,而勾选控件相对它们的"哥哥"节点的TextView是不同的,这样就可以先定位至"哥哥"节点,在根据相对位置,定位到指定的控件节点

在xpath中提供了多种轴方法,其中following-sibling可实现此功能 2ef37dd499d585fa0717dd1e1a6cd386.png

如我们要定位"画好一个封闭的圆"后面跟着的第二个RelativeLayout,具体写法如下:

//下面两种写法均可实现By.xpath("((//*[@text='画好一个封闭的圆'])[2]/following-sibling::android.widget.RelativeLayout)[2]")By.xpath("((//*[@text='画好一个封闭的圆'])[2]/following-sibling::*[@class='android.widget.RelativeLayout'])[2]")

很多控件都是有text属性的,但是appium是不支持直接对text进行定位的,而在实际工作中,我们经常会拿text进行定位,这就要归功于xpath了,通过对xpath语法的封装,我们就可以自定义一个根据text定位元素的方法来:

81fa3cb0d6a8aac9bb519eacb67d7aea.png 

public By ByText(String text){        return By.xpath("//*[@text='"+ text + "']");    }appiumdriver.findElement(ByText("关注"));

另外,需要定位Toast弹框时,有且仅有通过xpath的方式来实现:有时候我们进行某个操作后会弹出消息提示,例如点击某个按钮或下拉刷新后可能会出现类似"刷新成功"的提示语,然后几秒后消失;


42d18423c387abf572840aeb7fc6f0f1.gif 

弹出的消息很可能是Android系统自带的ToastToast在弹出的时候会在当前界面出现节点android.widget.Toast,随着消息的消失而消失;这个时候我们如果需要定位这个弹出消息,对其进行测试的话,就可以使用定位xpath方式了。

System.out.println( appiumdriver.findElementByXPath("//*[@class='android.widget.Toast']").getText());

结果:

 e6407688a7c615206d7d30c472bd9186.png 

更多xpath介绍可参考博客:推开Web自动化的大门到达“犯罪现象”-侦破selenium架构、环境安装及常用元素定位方法或W3C:https://www.w3school.com.cn/xpath/xpath_syntax.asp

2.23 通过id定位(每个元素原则上都有自己的唯一id值)-重要

学过web自动化的同学知道,在HTML中元素是有自己的id的,在移动端,元素依然有自己的id值,只不过名字叫做resource-id,如下:

970260ce7f8d88b06cf985100c4ccc2d.png

注: 我们看到id的值很长,其实实际使用只需要取斜杠/后面的部分就可以了,如下:

By.id("statusTitle")
2.24 通过accessibilityId定位(实则就是android的content-desc)-偶尔用到

在移动端自动化中有个特殊的定位方式就是根据accessibilityId定位,在dom中表现就是属性content-desc的值,如果Android中的content-desc中写入了值,便可以通过其进行定位:

 021df794a3309ffcff38bb55b63710c0.png

这里比较尴尬。。。由于研发经常偷懒不写,找了半天也没能找到例子,大家知道用法就好~另外要注意的是如果要写成"By.xxx"的形式,需要使用MobileBy

MobileBy.AccessibilityId("AccessibilityId");appiumdriver.findElementByAccessibilityId("AccessibilityId");
2.25 通过android uiautomator定位(相当于使用 UiAutomator Api 去递归地搜索元素(Android 专属))-高级用法

有时候我们需要对界面进行一定的操作方式后才能找到我们想要的元素,比如滑动列表进行查找等,这个时候就可以借助于android uiautomator了这里利用模拟器中的API Demo做演示,进入APIDemo中Views,然后滑屏寻找“Popup Menu”进行点击操作

08202ddf08896f65a94068b61ebb4e54.png

可以利用Android的UIAutomator进行滑屏操作,这时候需要使用AndroidDriver,另外定位元素可以使用UiScrollable

0cab2d8a9705ccf49c33fb585744e0f7.png

在官网的uiautomator UiSelector中有用ruby写的实例,不过定位方式是一致的,可以直接借鉴至java代码中

607c05b5a4a6b0dbaa7d7e872f7c9b5f.png

这里大概定位的方法就是,先用new UiSelector().scrollable(true).instance(0)判断是否可以滑动,找到ListView,然后用scrollIntoView(new UiSelector().text("WebView").instance(0)滑动找到对应定位属性的元素。

driver.findElementByXPath("//*[@text='Views']").click();((AndroidDriver)driver).                findElementByAndroidUIAutomator                 ("new UiScrollable(new UiSelector().scrollable(true).instance(0)).scrollIntoView(new UiSelector().text(\"Popup Menu\").instance(0))")                 .click();

在实际运行中,AndroidUIAutomator偶尔有定位失败的情况,可能在定位元素是位置会产生一点偏差,这里稍加改造避免这种偶发性失败;

改造方法:滑屏寻找元素时会先滑屏至待查元素的附近,这时元素已处于页面可见范围内,对元素操作可以重新定位操作,例如点击操作可以利用Xpath的方法重新定位后再click().

    By departmentName = MobileBy.AndroidUIAutomator(                "new UiScrollable(new UiSelector().scrollable(true).instance(0))." +                        "scrollIntoView(new UiSelector().text(\""+ departName +"\").instance(0))");        find(departmentName);//        click(departmentName); 原来直接操作滑动查找的元素结果        click(ByText(departName));//现在利用xpath重新定位确认后再操作,成功率大大提升

运行效果演示:

 b44f442acfdebec25e946cedffc273cb.gif

3. 定位逻辑的区别

59959262407a5418f7ed94fff6a5c6f4.png

在之前的一篇文章中我们介绍过appium底层的使用了各种引擎,可在文章末尾往期回顾第一条点击查看。

先简单看如下图:

 f51e35647a0cdd568c51ce541aee185f.png 

我们现在用的最新的版本优先支持的就是uiautomator2,如果你使用的是相对较前的版本,可能支持的是uiautomator,那么这两个引擎对于以上介绍的定位有什么影响呢?来看源码:

我们现在用的最新的版本优先支持的就是uiautomator2,如果你使用的是相对较前的版本,可能支持的是uiautomator,那么这两个引擎对于以上介绍的定位有什么影响呢?来看源码:

  • Uiautomator源码

 a0c4fc40bf8e2409ab03fdd167426b73.png

以id定位为例,在Uiautomator的源码可见其对id定位要更为宽泛,当我们使用By.id的时候,会同时去匹配resourceIdaccessibility idid

  • Uiautomator2源码

0e794bf918ec0c1fe11cb24447245589.png

在Uiautomator2中,将id的定位进行了细分,对应不同的id进行判断后再操作,因此在使用Uiautomator2的时候我们的写法要更为严谨

4. 推荐学习

59959262407a5418f7ed94fff6a5c6f4.png

appium官方说明文档:http://appium.io/docs/en/writing-running-appium/finding-elements/http://appium.io/docs/en/commands/element/find-elements/

Uiautomator2源码路径:

https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/FindElement.java

本文为霍格沃兹测试学院优秀学员的课程学习笔记,对于想系统进阶提升测试开发技能的同学,推荐霍格沃兹测试学院出品的 《测试开发从入门到高级实战》系统进阶班课程。

4 个月由浅入深,强化集训,测试大咖思寒领衔亲授,通过 8+ 企业级项目实战演练,带你一站式掌握 BAT 测试开发工程师必备核心技能(对标阿里巴巴P6+,挑战年薪50W+)!学员直推 BAT 名企测试经理,普遍涨薪 50%+!

0a3e8a4c367fc9d5f6800195b70a7c61.png

bd55bd8c4b6392f66a5cc34d45d2b5be.gif

a078c57046012815540d730c69b2293f.png”干货 | 掌握Selenium元素定位,解决...“

阅读

f23f8e3c515d1daa92c1994747a62889.png”基于Junit4,利用xUnit框架让你的测试...“

阅读

13a7813e6efd26ff2eff96c6e8656773.png”利器 | AppCrawler自动遍历与实践    ...“

阅读

ef5b20605101a28d3d60568924f61dcb.png关注我们

测试开发实战干货 | BAT 内推职位 | 大咖公开课

42c036c4396262bc88c206406da959aa.gif戳“阅读原文”, 来提升自己的核心竞争力吧!
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值