在上一章中我们有简要的介绍了事件源是怎么一回事,但是并没有进行详细的描述。那么往下的这几个小节我们就需要把这方面的知识给补充完整。
这一节我们先主要围绕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的主要作用:
最后放到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框架相关的事件源:
最后我们看下处理网络事件所涉及到的所有关键类的类关系图,先做初步了解以有个初步概念,往下的小节我们将会进行更详细的分析:
图6-1-3 事件处理关键类关系图
根据上一章的分析,Monkey类的runMonkeyCycles方法会循环执行,每循环一次都会调用事件源mEventSource的getNextEvent方法来获得一个事件,在这里就是MonkeySourceNetwork事件源的getNextEvent方法,然后调用该事件的injectEvent方法来处理相应的事件注入请求。那么我们结合上面的关键类关系图看每个类之间的关系及交互是怎么样的:
注:更多文章请关注公众号:techgogogo或个人博客http://techgogogo.com。当然,也非常欢迎您直接微信(zhubaitian1)勾搭。本文由天地会珠海分舵原创。转载请自觉,是否投诉维权看心情。