第5章7节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 循环获取并执行事件 - runMonkeyCycles(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文“寻求合作伙伴编写《深入理解 MonkeyRunner》书籍“。但因为诸多原因,没有如愿。所以这里把草稿分享出来,所以错误在所难免。有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息。

Monkey启动之后需要在整个MonkeyRunner的测试生命周期中提供服务,也就是说,一旦我们调用monkeyrunner命令来执行指定的测试脚本的时候,只要monkeyrunner还没有退出,那么Monkey就会一直提供服务,一直监听获取并处理从MonkeyRunner测试脚本发送过来的相应的命令。那么要实现这种一直监听的处理我们很自然就想到需要用一个循环来处理了。其实Monkey也不例外,我们往下会分析它是怎么通过一个循环来从事件源获取一个命令事件,然后调用该事件的注入方法进行事件注入的。当然,因为我们现在还没有分析学习到事件和事件源,所以本小节你只会看到Monkey是怎么通过它们提供的接口方法来获取事件和注入事件的,在下一章你就会看到整个事件究竟是怎么处理的了。

下面我们继续分析Monkey的run方法调用的最后一个重要方法,runMonkeyCycles。

代码5-7-1 Monkey - run调用runMonkeyCycles

 425     /**
 426      * Run the command!
 427      *
 428      * @param args The command-line arguments
 429      * @return Returns a posix-style result code. 0 for no error.
 430      */
 431     private int run(String[] args) {
 440         mCount = 1000;
    ... //省略初始化事件源等代码
 549         try {
 550             crashedAtCycle = runMonkeyCycles();
 551         }
    ...
}

当事件源初始化好后run方法就会去调用runMonkeyCycles方法不断的从命令队列中读取事件来执行。

代码5-7-2 Monkey - runMonkeyCycles

 915     /**
 916      * Run mCount cycles and see if we hit any crashers.
 917      * <p>
 918      * TODO: Meta state on keys
 919      *
 920      * @return Returns the last cycle which executed. If the value == mCount, no
 921      *         errors detected.
 922      */
 923     private int runMonkeyCycles() {
 924         int eventCounter = 0;
 925         int cycleCounter = 0;
        ...
     930         // TO DO : The count should apply to each of the script file.
 931         while (!systemCrashed && cycleCounter < mCount) {
            ...
1012             MonkeyEvent ev = mEventSource.getNextEvent();
1013             if (ev != null) {
1014                 int injectCode = ev.injectEvent(mWm,
                                        mAm, mVerbose);
            ...
1034                 // Don‘t count throttling as an event.
1035                 if (!(ev instanceof MonkeyThrottleEvent)) {
1036                     eventCounter++;
1037                     if (mCountEvents) {
1038                         cycleCounter++;
1039                     }
1040                 }
             }
    ...
}

这段代码做的主要事情就是通过931行的一个while循环去不停调用事件源提供的方法getNextEvent来获得一个事件,然后调用事件的injectEvent方法来注入事件,比如注入按键事件来模拟用户按下某个按键。往下我们对这段代码做相应的分析:

  • 第931行: 这里是一个while循环用来不停的从事件队列中读取事件。如果执行的事件没有导致系统崩溃,且当前循环次数少于mCount这个变量的话,整个循环都会继续。这里需要注意mCount这个成员变量,如果用户在执行monkey命令的时候没有指定”COUNT”这个运行次数的话,将会默认在Monkey的run这个方法中初始化成1000,请看” 代码5-9-1 Monkey - run调用runMonkeyCycles方法”第440行:” 440 this.mCount = 1000;”。
  • 第1012行: 从事件队列获取一个事件,注意对于网络事件会一直等到有事件才会返回。下一章会对事件源的getNextEvent的实现进行详尽的分析。
  • 第1014行:注入获取到的事件到系统来完成请求。下一章会对MonkeyEvent的injectEvent方法进行详尽的分析。

——— 未完待续———



作者:天地会珠海分舵

微信公众号:TechGoGoGo

微博:http://weibo.com/techgogogo

CSDN:http://blog.csdn.net/zhubaitian

时间: 2024-08-24 20:16:56

第5章7节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 循环获取并执行事件 - runMonkeyCycles(原创)的相关文章

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

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

第5章3节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 启动脚本(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 本节我们先看下Monkey是怎么启动起来的.在今后分析到MonkeyRunner的原理的时候我们会看到客户端是通过ADB往Android目标测试机器发送一个"monkey -port 12345"的

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

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

第4章2节《MonkeyRunner源码剖析》ADB协议及服务: ADB服务SERVICES.TXT翻译参考(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. ADB服务器端在接受到ADB客户端发送过来的命令后会进行相应的处理,如果是主机服务就在ADB服务器内部进行处理,如果是本地服务就会发送给Android目标机器端的adbd守护进程进行处理. 因为ADB相关的源代码不在我

第5章4节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 命令行参数解析(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 设置好Monkey的CLASSPATH环境变量以指定"/system/framework /framework/monkey.jar"后,/system/bin/monkey这个shell脚本就会通

第5章1节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 官方简介(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 在MonkeyRunner的框架中,Monkey是作为一个服务来接受来自MonkeyRunner客户端发送过来的命令,然后针对每条命令请求进行相应的处理,所以它并不是作为一个随机压力测试的工具来运作.本书中的Monke

第5章6节《MonkeyRunner源码剖析》Monkey原理分析-启动运行: 初始化事件源(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 事件源代表要注入系统的命令事件数据是从哪里过来的.这一小节我们不会对事件源的实现进行深入的分析,因为下一章会做这个事情.这里大家对事件源有个基本概念就足够了. 对Monkey来说,事件的来源可以有多个地方,比如我们用它

第4章4节《MonkeyRunner源码剖析》ADB协议及服务: ADB命令行客户端使用简介(原创)

天地会珠海分舵注:本来这一系列是准备出一本书的,详情请见早前博文"寻求合作伙伴编写<深入理解 MonkeyRunner>书籍".但因为诸多原因,没有如愿.所以这里把草稿分享出来,所以错误在所难免.有需要的就参考下吧,转发的话还请保留每篇文章结尾的出处等信息. 从前面几个小节我们知道ADB命令行客户端是存在与主机端的一个命令,用户可以使用该命令来发送服务请求到ADB服务器,ADB服务器再判断该服务请求是主机服务请求还是本地服务请求来决定是否应该将请求传送给远程adbd守护进程

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

本章我们重点围绕处理网络过来的命令的MonkeySourceNetwork这个事件源来阐述学习Monkey是如何处理MonkeyRunner过来的命令的.以下总结下MonkeyRunner从启动Monkey开始到如何处理完成一个命令的流程总结如下: MonkeyRunner通过ADB发送shell命令"monkey --port 12345"来启动Monkey Monkey启动运行进入runMonkeyCycles方法来循环获取并执行事件 runMonkeyCycles方法首先根据启动