第6章1节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览

在上一章中我们有简要的介绍了事件源是怎么一回事,但是并没有进行详细的描述。那么往下的这几个小节我们就需要把这方面的知识给补充完整。

这一节我们先主要围绕MonkeySourceNetwork这个事件源来学习事件源的框架结构。首先,要理解事件源,必须先搞清楚几个问题:

  • 事件从哪里来?

Monkey的事件来源有多个方面,但是作为MonkeyRunner框架的一部分,它的事件来源主要是来自MonkeyRunner通过网络Socket(USB/TCP协议)发送过来的命令字串。MonkeySourceNetwork这个事件源类就是专门处理这些请求的。MonkeySourceNetwork会在初始化的过程中建立一个ServerSocket来供客户端连接,Socket的端口就是MonkeyRunner通过ADB shell发送给Android目标机器的启动monkey的命令“monkey –port 12345”中的12345。

556     public MonkeySourceNetwork(int port) throws IOException {
557         // Only bind this to local host.  This means that you can only
558         // talk to the monkey locally, or though adb port forwarding.
559         serverSocket = new ServerSocket(port,
560                                         0, // default backlog
561                                         InetAddress.getLocalHost());
562     }

代码6-1-1 MonkeySourceNetwork - 构造函数

  • 来自网络的事件命令字串如何转换成事件?

来自网络的字串是不能直接使用的,Monkey必须把该命令字串进行解析,在必要的时候转换成对应的Monkey事件,这个过程在Monkey中称为命令翻译。MonkeySourceNetwork一旦从MonkeyRunner客户端获取一个字串命令,它就会根据其内部成员COMMAND_MAP这个“字串命令 - 命令翻译类对象”的映射表,检索到该命令字串对应的翻译类对象,然后就会调用它的命令翻译方法来把该字串命令翻译成对应的MonkeyEvent事件,这里说的MonkeyEvent是所有具体事件如MonkeyFlipEvent的父类。以下代码就是COMMAND_MAP在MonkeySourceNetwork类中的实现:

449     private static final Map<String, MonkeyCommand> COMMAND_MAP = new HashMap<String, MonkeyCommand>();
450
451     static {
452         // Add in all the commands we support
453         COMMAND_MAP.put("flip", new FlipCommand());
454         COMMAND_MAP.put("touch", new TouchCommand());
455         COMMAND_MAP.put("trackball", new TrackballCommand());
456         COMMAND_MAP.put("key", new KeyCommand());
457         COMMAND_MAP.put("sleep", new SleepCommand());
458         COMMAND_MAP.put("wake", new WakeCommand());
459         COMMAND_MAP.put("tap", new TapCommand());
460         COMMAND_MAP.put("press", new PressCommand());
461         COMMAND_MAP.put("type", new TypeCommand());
462         COMMAND_MAP.put("listvar", new MonkeySourceNetworkVars.ListVarCommand());
463         COMMAND_MAP.put("getvar", new MonkeySourceNetworkVars.GetVarCommand());
464         COMMAND_MAP.put("listviews", new MonkeySourceNetworkViews.ListViewsCommand());
465         COMMAND_MAP.put("queryview", new MonkeySourceNetworkViews.QueryViewCommand());
466         COMMAND_MAP.put("getrootview", new MonkeySourceNetworkViews.GetRootViewCommand());
467         COMMAND_MAP.put("getviewswithtext",
468                         new MonkeySourceNetworkViews.GetViewsWithTextCommand());
469         COMMAND_MAP.put("deferreturn", new DeferReturnCommand());
470     }

代码6-1-2 MonkeySourceNetwork - COMMAND_MAP

它的键是String类型的字串,代表的是从网络过来的命令字串;它的值是MonkeyCommand的实例,代表的是负责将该命令字串翻译成对应事件的类实例,但要注意的是并不是所有的命令都会生成对应的事件对象并放到事件队列里等待执行,有些命令会在翻译的过程中直接处理返回的。往下描述MonkeyEvent事件的时候会有更详尽的描述。

COMMAND_MAP里面的键记录的只是命令字串,没有包含对应的参数,以下列出Monkey支持的从网络过来的所有命令和对应的参数:


命令字串格式


MR是否支持


注释


touch down x y



x代表x坐标,y代表y坐标


touch up x y



同上


touch move x y



同上


tap x y



同上


press name



Name代表按键名,如"MENU", "HOME", "SEARCH"等


key down name



同上


key up name



同上


getvar name



name 代表属性名


listvar



type line



line 代表输入字串


wake



listViews



queryview



getRootView



getViewWithText



done



测试完成,Monkey收到命令后会停止Socket监听


quit



测试请求退出,Monkey收到后不会停止Socket监听,


flip open



MonkeyRunner不支持发送这两个命令


flip close



trackball dx dy



deferredReturn



这个命令比较特别,做的事情是等待一个命令完成然后执行另外一个命令。但是在MonkeyRunner框架中并没有支持。


注: MR代表MonkeyRunner

表6-1-1 命令字串参照表

  • 事件要到那里去?

每个事件源处理类都维护着一个自己的事件队列, 在Monkey中叫做CommandQueue,里面装的是每个具体的MonkeyEvent事件。当来自网络的字串命令被翻译成对应的MonkeyEvent的时候就会把它追加到事件队列里面

了解了以上几个问题,我们脑袋里就有MonkeySourceNetwork这个事件源的基本的概念了,以下简要总结下MonkeySourceNetwork的主要作用:

  • 从网络获取命令字串
  • 使用相应的命令翻译类把命令字串翻译成MonkeyEvent事件并把事件放入到命令队列CommandQueue

最后放到CommandQueue里面做什么呢?当然是给Monkey类的runMonkeyCycles方法去消费事件了,还记得它每个循环都会调用mEventSource.getNextEvent去获取一个事件然后进行事件注入吧?具体请查看上一章第7节runMonkeyCycles方法的介绍。

下图示例显示了从MonkeyRunner测试脚本调用MonkeyDevice的type方法来往系统窗口输入字串开始,到该type命令字串由网络事件源获取到并翻译成对应的事件,加入事件队列,最后取出事件进行事件注入的流程。其中红色方框圈住部分就是刚才描述的网络事件源类MonkeySourceNetwork主要做的事情。

图6-1-1 事件处理示意图

下面我们看下事件源类的族谱,这里只有我们需要重点描述的MonkeySourceNetwork事件源会把重要的成员方法和成员变量显示出来,但这不代表其他事件源就只有一个getNextEvent方法,只是它们在MonkeyRunner框架中不起什么作用,所以省略了而已:

图6-1-2 事件源族谱

MonkeyEventSource定义了所有事件源的接口,里面定义了所有实现该接口的事件源都必须实现getNextEvent的方法来从各自对应的事件队列CommandQueue中取事件,其实这里CommandQueue更应该叫做EventQueue,这样会更容易理解,因为它里面装的都是事件。这个设计很好的体现了面向对象的多态特性,因为这样的话在上一章描述的runMonkeyCycles方法中就只需要调用MonkeyEventSource接口实例mEventSource的getNextEvent方法就能取得一个事件来执行了,而它根本不需要知道调用的是哪个事件源来获得事件的,因为mEventSource指定的就是所有事件的顶层接口MonkeyEventSource。以下重点描述下跟MonkeyRunner框架相关的事件源:

  • MonkeySourceNetworkVars和MonkeySourceNetworkViews并不会直接实现MonkeyEventSource接口,其实它们的事件都是通过MonkeySourceNetwork来从网络获得的,MonkeySourceNetwork在获得网络过来的事件后,如果发现它们是根获取系统属性和获取界面控件有关,那么就会分别调用MonkeySourceNetworkVars和MonkeySourceNetworkViews里面对应的MonkeyCommand内部类的方法translateCommand来把网络过来的字串事件命令翻译成对应的MonkeyEvent
  • 这里MonkeySourceNetwork这个事件源是我们分析的重点,它持有成员变量commandQueue,里面装的就是所有将要执行的MonkeyEvent事件
  • MonkeySourceNetwork内部建立了一个网络过来的命令字串到对应的MonkeyCommand实例的映射,MonkeyCommand是专门用来将该命令字串翻译成MonkeyEvent事件的
  • MonkeySourceNetwork在初始化时会建立一个ServerSocket来接收从网络过来的也就是MonkeyRunner客户端发送过来的字串命令,并调用自身的translateCommand方法来根据该命令名字在分发到其在CommandMap对应的MonkeyCommand来把它翻译成对应的事件

最后我们看下处理网络事件所涉及到的所有关键类的类关系图,先做初步了解以有个初步概念,往下的小节我们将会进行更详细的分析:

图6-1-3 事件处理关键类关系图

根据上一章的分析,Monkey类的runMonkeyCycles方法会循环执行,每循环一次都会调用事件源mEventSource的getNextEvent方法来获得一个事件,在这里就是MonkeySourceNetwork事件源的getNextEvent方法,然后调用该事件的injectEvent方法来处理相应的事件注入请求。那么我们结合上面的关键类关系图看每个类之间的关系及交互是怎么样的:

  • Monkey组合MonkeySourceNetwork: Monkey拥有私有成员mEventSource,该成员变量在MonkeyRunner请求”monkey --port 12345”启动monkey的过程中会初始化成MonkeySourceNetwork的实例,详情请看上一章的第6小节”初始化事件源”
  • MonkeySourceNetwork组合CommandQueueImpl: MonkeySourceNetwork维护着一个commandQueue的成员变量,这个成员变量是一个保存着所有等待执行的事件的一个队列,也就是我们经常提及的MonkeyEvent事件队列
  • CommandQueueImpl组合 MonkeyEvent: 事件队列里边包含有0到多个Monkey事件MonkeyEvent的实例。Monkey的runMonkeyCycles方法里面的每一个cycle都会去该命令队列中取出一条命令来执行

注:更多文章请关注公众号:techgogogo或个人博客http://techgogogo.com。当然,也非常欢迎您直接微信(zhubaitian1)勾搭。本文由天地会珠海分舵原创。转载请自觉,是否投诉维权看心情。

时间: 2024-12-29 04:48:39

第6章1节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览的相关文章

老李推荐: 第8章4节《MonkeyRunner源码剖析》MonkeyRunner启动运行过程-启动AndroidDebugBridge 1

老李推荐: 第8章4节<MonkeyRunner源码剖析>MonkeyRunner启动运行过程-启动AndroidDebugBridge 上一节我们看到在启动AndroidDebugBridge的过程中会调用其start方法,而该方法会做2个主要的事情: 715行startAdb:开启AndroidDebugBridge 722-723行:初始化android设备监控并启动DeviceMonitor设备监控线程. 其中第一点我们上一小节已经做了详尽分析了,那么我们往下就去分析下第2点. Dev

老李推荐:第6章5节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-事件

老李推荐:第6章5节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-事件 从网络过来的命令字串需要解析翻译出来,有些命令会在翻译好后直接执行然后返回,但有一大部分命令在翻译后需要转换成对应的事件,然后放入到命令队列里面等待执行.Monkey在取出一个事件执行的时候主要是执行其injectEvent方法来注入事件,而注入事件根据是否需要往系统注入事件分为两种: 需要通过系统服务往系统注入事件:如MonkeyKeyEvent事件会通过系统的InputManager往系

老李推荐:第6章3节《MonkeyRunner源码剖析》Monkey原理分析-事件源-事件源概览-命令翻译类

老李推荐:第6章3节<MonkeyRunner源码剖析>Monkey原理分析-事件源-事件源概览-命令翻译类 每个来自网络的字串命令都需要进行解析执行,只是有些是在解析的过程中直接执行了事,而有些是需要在解析后创建相应的事件类实例并添加到命令队列里面排队执行.负责这部分工作的就是命令翻译类.那么我们往下还是继续在MonkeySourceNetwork这个范畴中MonkeyCommand类是怎么一回事: 图6-3-1 MonkeyCommand族谱 图中间的MonkeyCommand是一个接口,

老李推荐: 第1章1节《MonkeyRunner源码剖析》概述:前言

老李推荐: 第1章1节<MonkeyRunner源码剖析>概述:前言 前言 相信大家做过安卓移动平台UI自动化开发的必然会用过,至少听过MonkeyRunner这个名字.MonkeyRunner是一个针对安卓平台的UI自动化测试框架,这个框架的其中一个但绝不是唯一的优点是支持用当今非常流行和高效的Python语言来进行脚本开发.同时,它相比Instrumentation框架或者基于Instrumentation的自动化测试框架最大的优点之一就是可以跨应用测试. 这本书不会有什么序言或者致谢什么

boost.asio源码剖析(三) ---- 流程分析

* 常见流程分析之一(Tcp异步连接) 我们用一个简单的demo分析Tcp异步连接的流程: 1 #include <iostream> 2 #include <boost/asio.hpp> 3 4 // 异步连接回调函数 5 void on_connect(boost::system::error_code ec) 6 { 7 if (ec) // 连接失败, 输出错误码 8 std::cout << "async connect error:"

第5章5节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 获取系统服务引用(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 上一节我们描述了monkey的命令处理入口函数run是如何调用optionProcess方法来解析命令行参数的.启动参数主要时去指导Monkey时怎么运行起来的,但Monkey作为MonkeyRunner框架的一部分,

第5章2节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 启动流程概览(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 每个应用都会有一个入口方法来供操作系统调用执行,Monkey这个应用的入口方法就是在Monkey.java这个类里面的,也就是说Monkey.java就是整个Monkey应用的入口类. Monkey作为一个命令行应用,

第2章1节《MonkeyRunner源码剖析》了解你的测试对象: NotePad应用简介(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 本书脚本相关的示例常会用到Android SDK自带的NotePad这个应用,所以这里很有必要去描述下这个应用.其实去熟悉这个应用最好的办法是自己去把它安装上去你的目标安卓系统,然后亲自去玩一下它的各种功能.因为它只是

第1章1节《MonkeyRunner源码剖析》概述:前言(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 前言 相信大家做过安卓移动平台UI自动化开发的必然会用过,至少听过MonkeyRunner这个名字.MonkeyRunner是一个针对安卓平台的UI自动化测试框架,这个框架的其中一个但绝不是唯一的优点是支持用当今非常流

第1章2节《MonkeyRunner源码剖析》概述:边界(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 边界 怎么样才算分析清楚一个事物的原理是什么呢?就以前面提到的<LINUX内核源代码情景分析>为例子,分析到什么程度算是把Linux内核给分析清楚呢?一个操作系统内核的原理无非是要描述清楚这几个核心是如何工