J 今天的主题是讲一下在使用过程中遇到的一个问题,如何在UiAutomator2.0中使用Xpath定位元素?

  背景

  现在的app在打包成apk的时候都是有加固处理的,各种混淆加固,所以已经破坏了或扰乱了原本的代码变量命名形式,这就给我们要基于界面来做自动化测试带来了灾难性的阻碍,因为那些混淆过的id是不固定的,下一次再出个新版本,这一切都变了,所以这就没办法用id来定位混淆过的app元素,那还有什么好的方法吗?还记得Web自动化测试中神乎其技的xpath吗?不管什么元素都可以用它定位出来,所以我就想在UiAutomator2.0中也使用它来定位混淆的app元素,这要如何操作?UiAutomator2.0的API中并没有给出xpath这种方式,那我们只能自己去写一个了。

  思路

  参考UI Automator Viewer中抓取到的结构层次,不能用resource-id,又要体现出层次关系,那就只能是class属性了,这里的class可以对应web xpath中的标签,使用业界统一的斜杠/来保持层次,那么最原始状态下的xpath大概就是这个样子了:

  android.view.ViewGroup/android.widget.ImageView

  再加上下标

  android.view.ViewGroup[2]/android.widget.ImageView[0]

  

  xpath的格式定义出来了之后,我们就开始一层一层去遍历,很简单通过斜杠/来分隔出一个class数组,然后依次去查找这些class对应的元素,通过父子关系拼接起来,直到最后一个class,存在就返回对应的对象,不存在就返回null。

  由于时间关系,这一次就是初探,只实现了这种绝对路径(/)下的定位,其实要想完整完成这个功能,还需要支持相对路径(//)的定位,以及各种属性的组合定位,其实基于这个版本上面改改也不远了,这就留给有兴趣的童鞋去完成吧。

  实现

  1、首先要实现根据class或其他属性去找到某个元素的子元素,我这里实现了支持传入各种参数,代码如下:

  public static UiObject2 getChild(Object root, Mapparams) { if (params == null || !params.containsKey(class)) {

  log.e([Error]参数错误: 为空或未包含[class]key); return null;

  } String clazz = params.get(class); String className = clazz;

  int index = 0; if (clazz.endsWith(]) clazz.contains([)) { //有下标

  className = clazz.substring(0, clazz.lastIndexOf([)); String num = clazz.substring(clazz.lastIndexOf([) + 1, clazz.lastIndexOf(]));

  index = num != null !.equals(num) ? Integer.parseInt(num) : index;

  }

  ListchildList = null; if (root instanceof UiObject2) {

  childList = ((UiObject2) root).getChildren();

  } else {

  childList = hasObjects(By.clazz(className)) ? mDevice.findObjects(By.clazz(className)) : null;

  }

  ListtempList = new ArrayList(); if (childList != null !childList.isEmpty()) { for (UiObject2 child : childList) {

  boolean isMatch = child.getClassName().equals(className); if (params.containsKey(pkg)) {

  isMatch = isMatch child.getApplicationPackage().equals(params.get(pkg));

  } if (params.containsKey(text)) {

  isMatch = isMatch child.getText().equals(params.get(text));

  } if (params.containsKey(desc)) {

  isMatch = isMatch child.getContentDescription().equals(params.get(desc));

  } if (isMatch) {

  tempList.add(child);

  }

  }

  } if(tempList.isEmpty()) { return null;

  } if (index = tempList.size()) {

  log.e(String.format([Error]查找class[%s] 下标[%d]越界[%d], clazz, index, tempList.size())); return null;

  } return tempList.get(index);

  }

  2、再写一个通过class获取子元素的简单实现,因为这种方式用的多:

  public static UiObject2 getChild(Object root, String clazz) { Mapparams = new HashMap();

  params.put(class, clazz); return getChild(root, params);

  }

  3、加入解析xpath表达式的部分,将解析和查找整个过程连起来:

  public static UiObject2 findObjectByXpath(UiObject2 root, String xpath) { if (xpath == null .equals(xpath)) {

  log.e([Error]xpath expression[ + xpath + ] is invalid); return null;

  } String[] xpaths = null; if (xpath.contains(/)) {

  xpaths = xpath.split(/);

  } else {

  xpaths = new String[]{xpath};

  }

  UiObject2 preNode = root; for (String path : xpaths) {

  preNode = getChild(preNode, path); if (preNode == null) { //log.e(String.format(按xpath[%s]查找元素失败, 未找到class[%s]对应的节点, xpath, path));

  break;

  }

  } return preNode;

  }

  4、使用演示:

  String commentXpath = android.widget.LinearLayout/android.widget.LinearLayout/android.widget.TextView[0];

  UiObject2 commentView = findObjectByXpath(root, commentXpath);

初探UiAutomator2.0中使用Xpath定位元素的更多相关文章

  1. 『心善渊』Selenium3.0基础 — 6、Selenium中使用XPath定位元素

    目录 1.Selenium中使用XPath查找元素 (1)XPath通过id,name,class属性定位 (2)XPath通过标签中的其他属性定位 (3)XPath层级定位 (4)XPath索引定位 ...

  2. 使用python处理selenium中的xpath定位元素的模糊匹配问题

    # 用contains,寻找页面中style属性值包含有sp.gif这个关键字的所有div元素,其中@后面可以跟该元素任意的属性名. self.driver.find_element_by_xpath ...

  3. Selenium-Python学习——通过XPath定位元素

    用Xpath定位元素的方法总是记不住,经常要翻出各种文档链接参考,干脆把需要用到的内容整到这个笔记中方便查找. Xpath是在XML文档中定位节点的语言.使用 XPath 的主要原因之一是当想要查找的 ...

  4. Robot Framework与Web界面自动化测试学习笔记:利用xpath定位元素

    在rf中,利用selinum2的关键字进行用例编写时,很多关键字的参数是html元素的定位标识. 最简单的方式,是通过id 或name来描述元素定位信息,如 click  button    id=l ...

  5. 【appium】根据xpath定位元素

    1. 背景 本文尝试使用的试验对象是SDK自带的NotePad应用实例,假设已经有两个Notes分别是“note1”和“note2”添加到Notepad上面,我们要做的就是尝试用xpath的方法来定位 ...

  6. Python selenium PO By.XPATH定位元素报错

    Python selenium PO  By.XPATH定位元素报错 如下代码经常报错: # 首页的“新建投放计划”按钮 new_ads_plan = (By.XPATH, "//*[tex ...

  7. Xpath定位元素-一个例子

    前几天在群里面解决的问题,记录下来和大家分享 需要定位这个股份制企业 方法: # 首先需要单击下拉框弹出企业性质的下拉选项:然后用过Xpath定位元素 driver.find.element_by_c ...

  8. selenium中的xpath用法,使用xpath定位元素

    xpath路径选择器定位元素 1. xpath: 使用路径表达式来定位xml或者html中文档中选取节点.在 XPath 中,有七种类型的节点:元素.属性.文本.命名空间.处理指令.注释以及文档节点( ...

  9. Selenium3自动化测试【18】XPath定位元素(2)

    层级与属性结合定位 如果被定为的元素,无法通过自身属性来唯一标识自己,此时可以考虑借助上级元素来定位自己.举生活中的例子,一个婴儿刚出生,还没有姓名与身份证号,此时给婴儿进行检查时往往会标注为&quo ...

随机推荐

  1. HYSBZ 3676 回文串 (回文树)

    3676: [Apio2014]回文串 Time Limit: 20 Sec  Memory Limit: 128 MB Submit: 1680  Solved: 707 [Submit][Stat ...

  2. python基础之类的继承与派生、组合、接口与归一化设计、抽象类、子类中调用父类方法

    一.什么是继承 继承是一种创建新的类的方式,新建的类可以继承自一个或者多个父类,原始类称为基类或超类,新建的类称为派生类或子类. 派生:子类继承了父类的属性,然后衍生出自己新的属性,如果子类衍生出的新 ...

  3. Pandas 之 过滤DateFrame中所有小于0的值并替换

    Outline 前几天,数据清洗时有用到pandas去过滤大量数据中的“负值”: 把过滤出来的“负值”替换为“NaN”或者指定的值. 故做个小记录. 读取CSV文件 代码: import pandas ...

  4. MongoDB-4: 查询(二-数组、内嵌文档)

    一.简介 我们上一篇介绍了db.collection.find()可以实现根据条件查询和指定使用投影运算符返回的字段省略此参数返回匹配文档中的所有字段,我们今天介绍了对数组和内嵌文档的查询操作,尤其是 ...

  5. SQL逻辑查询语句 执行顺序 语法顺序

    SELECT语句语法顺序 SELECT DISTINCT <select_list> FROM <left_table> <join_type> JOIN < ...

  6. vim 设置 颜色值

    编辑~/.vimrc文件,添加 set t_Co=8 t_Co即terminal Color之意 注意,将 t_Co 设置为256 (或8以外的所有值) 时,mark 的显示不是很正常.

  7. CentOS7编译安装MariaDB

    一.环境信息: 操作系统版本:CentOS Linux release 7.3.1611 (Core) 内核版本:3.10.0-514.el7.x86_64 MariaDB版本:mariadb-10. ...

  8. SignalTap II进阶学习

    1. 多级触发 有时候我们可能需要利用多个信号(最多10个)依次触发后,观察特定的值.这这时候我们可以增加触发条件来满足我们的需求. 多个触发信号如上图所示,只有依次发生SW[0]上升沿. SW[1] ...

  9. rem根元素的定义规则

    <script> !function(a) { function b() { var b = g.getBoundingClientRect().width; var winpix=1; ...

  10. 用python的turtle画分形树

    由于分形树具有对称性,自相似性,所以我们可以用递归来完成绘制.只要确定开始树枝长.每层树枝的减短长度和树枝分叉的角度,我们就可以把分形树画出来啦!! 代码如下: # -*- coding: utf-8 ...