老李推荐: 第14章2节《MonkeyRunner源码剖析》 HierarchyViewer实现原理-HierarchyViewer架构概述
老李推荐: 第14章2节《MonkeyRunner源码剖析》 HierarchyViewer实现原理-HierarchyViewer架构概述
HierarchyViewer库的引入让MonkeyRunner可以以面向控件的方式进行脚本编写,那么它是怎么做到这一点的呢?它的实现原理又是怎么样的呢?在这一小节开始我们就会开始一起揭开它的神秘面纱。
HierarchyViewer这个框架的工作需要ViewServer的协同工作,它们是以CS(Client-Server)架构来协同工作的。我们在上一章已经学习了如何让做为Server端的ViewServer为我们提供获取控件列表的服务,那么这一章我们主要分析的就是做为Client端的HierarchyViewer是如何实现驱动目标机器的ViewServer服务器来获取控件列表等相关工作的。
从我们编写MonkeyRunner测试脚本的角度来看,我们一般会这样子使用HierarchyViewer:
1 device = MonkeyRunner.waitForConnection()
2 viewer = device.getHierarchyViewer()
3 view = viewer.findViewById("id/button") #Return a ViewNode
4 p = viewer.getAbsoluteCenterOfView(view)
5 device.touch(p.x,p.y,MonkeyDevice.DOWN_AND_UP)
代码14-2-1 通过控件点击按钮
从以上的代码我们可以看到整个使用HierarchyViewer的流程是非常的简单明了的:
- 第一步: 通过MonkeyDevice的实例获得HierarchyViewer的实例
- 第二步: 使用HierarchyViewer的实例来通过控件的ID来获得代表控件的一个ViewNode实例
- 第三步: 最后就可以通过操作ViewNode实例暴露出来的各种控件属性来进行控件的操作了
我们最终通过ViewNode控件实例的属性来操作控件确实让我们用起来得心应手,但是暗藏在这些简单的脚本代码背后的实现原理其实并不会这么简单的。我们尝试把我们脑袋里面的疑问给列出来:
- 问题1: findViewById是从哪里根据控件ID获得我们需要的控件的?直接把ID发给ViewServer让它去查找吗?还是通过其他方法?
- 问题2 :MonkeyDevice的getHierarchyViewer方法究竟做了什么事情?
这两个问题的答案其实就是本章的重点,我们先带着这两个问题来往下学习,最后问题就会不言自明了。
往下我们先看下HierarchyViewer涉及的关键类的类图来了解下它的架构:

图11-2-1 HierarchyViewer类图
从上图我们可以看到整个框架其实涉及到了多个库,包括chimpchat,hierarchyviewerlib和ddmlib类,这里先简要描述下主要会用到每个库的哪些类,以及这些类在这里主要的作用是什么,其实很多我们在前面章节都已经接触过了,但以前可能没有去描述这些类跟HierarchyViewer是什么关系。
|
类名 |
库 |
作用 |
注释 |
|
HierarchyViewer |
chimpchat |
提供控件相关API给测试开发人员调用,如findViewById等 |
拥有ddmlib的Device类的一个实例,所以可以通过该实例来操作AdbHelper来跟ADB服务器进行通信 |
|
Device |
ddmlib |
在这里的主要作用是可以直接调用AdbHelper来往ADB服务器发送命令 |
|
|
AdbHelper |
ddmlib |
负责连接ADB服务器并向其发送命令 |
|
|
DeviceBridge |
hierarchyviewerlib |
相当于ViewServer在客户端的代理,两者的关系类似于adbd和adb服务器的关系; |
组织命令然后调用DeviceConnection的API把命令发送给ViewServer |
|
DeviceConnection |
hierarchyviewerlib |
连接ViewServer的Socket客户端,专门用来连接ViewServer,发送命令和接收命令返回 |
|
|
Window |
hierarchyviewerlib |
代表一个控件的窗口,它跟控件ViewNode是不一样的,它不会保存控件的所有属性,它主要的成员变量有:窗口标题mTitle;代表该窗口的哈希值 |
|
|
ViewNode |
hierarchyviewerlib |
代表了一个控件,拥有Window实例以及一个控件的所有属性信息 |
表格 11-2-1 HierarchyViewer关键类简述
往下我们描述下每个类之间是怎么互动的:
- HierarchyViewer依赖DeviceBridge: HierarchyViewer在初始化的时候会调用途中的setupViewServer这个私有方法来准备好目标机器端的ViewServer,期间它会调用DeviceBridge的一系列方法来完成这些事情,其中DeviceBridge是一个静态类,不需要初始化就能直接使用其内部公共方法
- DeviceBridge依赖DeviceConnection: DeviceBridge成员方法会初始化DeviceConnection对象来通过它把组织好的命令发送到ViewServer,比如DUMP命令.
- DeviceBridge依赖ViewNode: DeviceBridge在通过发送DUMP命令到ViewServer获得控件列表后,会把每个控件的信息都解析出来,然后建立一个由ViewNode控件实例组建起来的一棵控件树。
- DeviceBridge依赖Device: HierarchyViewer在调用DeviceBridge的setupViewServer去准备好ViewServer的时候会把它持有的Device实例传进去给DeviceBridge,DeviceBridge就是通过这个Device实例来组织相应的ADB命令来发送给ADB服务器来操作ViewServer的
- Device依赖AdbHelper: Device本身并不会连接ADB服务器的socket来发送命令,这些都是由AdbHelper来做的,我们在之前的章节已经见识过了
- ViewNode组合Window: 每个ViewNode都拥有一个Window成员变量。也就是说每个控件都应该是属于某一个窗口的,一个不属于某个窗口的控件是没有意义的,是不应该存在的
- HierarchyViewer使用ViewNode: 脚本在调用HierarchyViewer的findElementById的时候,会首先使用DeviceBridge来获取控件列表并建立控件树,最后会遍历由ViewNode实例组成的控件树来获取目标控件
HierarchyViewer架构到了这里就算描述的差不多了,往下我们会进入到实现细节去看下它的实现原理是怎样的。
老李推荐: 第14章2节《MonkeyRunner源码剖析》 HierarchyViewer实现原理-HierarchyViewer架构概述的更多相关文章
- 老李推荐:第6章8节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-小结
老李推荐:第6章8节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-小结 本章我们重点围绕处理网络过来的命令的MonkeySourceNetwork这个事 ...
- 老李推荐:第6章7节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-注入按键事件实例
老李推荐:第6章7节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-注入按键事件实例 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜 ...
- 老李推荐:第6章6节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-命令队列
老李推荐:第6章6节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-命令队列 事件源在获得字串命令并把它翻译成对应的MonkeyEvent事件后,会把这些 ...
- 老李推荐:第6章4节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-翻译命令字串
老李推荐:第6章4节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-翻译命令字串 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自 ...
- 老李推荐:第6章5节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-事件
老李推荐:第6章5节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-事件 从网络过来的命令字串需要解析翻译出来,有些命令会在翻译好后直接执行然后返回,但有 ...
- 老李推荐:第6章3节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-命令翻译类
老李推荐:第6章3节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-命令翻译类 每个来自网络的字串命令都需要进行解析执行,只是有些是在解析的过程中直接执行 ...
- 老李推荐:第6章2节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-获取命令字串
老李推荐:第6章2节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-获取命令字串 从上一节的描述可以知道,MonkeyRunner发送给Monkey的命令 ...
- 老李推荐:第5章7节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 循环获取并执行事件 - runMonkeyCycles
老李推荐:第5章7节<MonkeyRunner源码剖析>Monkey原理分析-启动运行: 循环获取并执行事件 - runMonkeyCycles poptest是国内唯一一家培养测试开 ...
- 老李推荐:第5章6节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 初始化事件源
老李推荐:第5章6节<MonkeyRunner源码剖析>Monkey原理分析-启动运行: 初始化事件源 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试 ...
- 老李推荐:第5章3节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 启动脚本
老李推荐:第5章3节<MonkeyRunner源码剖析>Monkey原理分析-启动运行: 启动脚本 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性 ...
随机推荐
- 发布自己的Angular2库初探
从去年年底开始使用ng2,遇到并解决或被虐了一些问题点,对其各种新特性与开发模式感觉还算舒服.还有的一个感想就是,要使用ng2还得先学习不少其他东西,比如TypeScript语法,比如ES6新特性,还 ...
- 【转】Netty系列之Netty是什么
Netty是什么 大概用Netty的,无论新手还是老手,都知道它是一个“网络通讯框架”.所谓框架,基本上都是一个作用:基于底层API,提供更便捷的编程模型.那么”通讯框架”到底做了什么事情呢?回答这个 ...
- Mr.聂 带你成为web开发大牛——入门篇(上)
作为一名IT届的后生,当初也经历过懵懂无知的实习期,对那种无力感深有体会.在这,希望能用我这几年的开发经验,让各位即将踏入或者刚刚踏入web开发领域的新人们少走些弯路.鉴于这是入门篇,下面我就从零为大 ...
- DAX/PowerBI系列 - 写在前面
今天讲的主角是: 不过,先上一个图--2017 Gartner商业智能和数据分析魔力象限 DAX关注这个玩意儿有好一段时间了,刚开始的时候(2014年?)是从Excel里面认识的.2014年当时公司用 ...
- wemall app商城源码中基于PHP的通用的树型类代码
wemall doraemon是Android客户端程序,服务端采用wemall微信商城,不对原商城做任何修改,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可随意定制修改.本文分享其中 ...
- OpenvSwitch Port Mirror in OpenStack Neutron
前言 最近使用搭建了一个基于VXLAN的OpenStack 环境,发现要去dump ovs interfaces的包其实还是蛮麻烦的, 经过多番努力,找到了如下的在openstack下网络环境的一些t ...
- 深度剖析Spark分布式执行原理
让代码分布式运行是所有分布式计算框架需要解决的最基本的问题. Spark是大数据领域中相当火热的计算框架,在大数据分析领域有一统江湖的趋势,网上对于Spark源码分析的文章有很多,但是介绍Spark如 ...
- 为什么用IP无法访问网站,域名可以访问?
我们访问网站都是通过域名进行访问的,偶尔会使用网站IP进行访问,如学校通常使用IP登录教务处,但很多的时候我们无法通过ip进行访问其他网站,这就涉及到服务器的问题了. 网站都是依托在服务器上面的,而服 ...
- Python 爬取qqmusic音乐url并批量下载
qqmusic上的音乐还是不少的,有些时候想要下载好听的音乐,但有每次在网页下载都是烦人的登录什么的.于是,来了个qqmusic的爬虫. 至少我觉得for循环爬虫,最核心的应该就是找到待爬元素所在ur ...
- HTML 表单常用的代码元素
表单: 将数据通过浏览器提交到服务器的媒介.<form action="" method="get/post" ></form> get ...