大家好,又见面了,我是全栈君。
在《MonkeyRunner源代码分析之与Android设备通讯方式》中。我们谈及到MonkeyRunner控制目标android设备有多种方法。当中之中的一个就是在目标机器启动一个monkey服务来监听指定的一个port,然后monkeyrunner再连接上这个port来发送命令。驱动monkey去完毕对应的工作。
当时我们仅仅分析了monkeyrunner这个client的代码是怎么实现这一点的,但没有谈monkey那边是怎样接受命令,接受到命令又是怎样处理的。
所以自己打开源代码看了一个晚上,大概有了概念。但今天网上搜索了下。发现已经有网友“chenjie”对monkey的源代码做过对应的分析了,并且文章写得很有概括性。应该是高手所为,果断花了2个积分下载下来。不敢独享。本想贴上来分享给大家。但发现pdf的文档直接拷贝上来会丢失掉图片,所以仅仅好贴上下载地址:http://download.csdn.net/download/zqilu/6884491
但文章主要是架构性得去描写叙述monkey是怎么工作的,依照我自己的习惯。我还是喜欢依照自己的思维和有目的性的去了解我想要的。在这里我想要的是搞清楚monkey是怎样处理monkeyrunner过来的命令的。
本文我们就先看下monkey的执行流程。
1. 执行环境设置
和monkeyrunner一样。monkey这个命令也是一个shell脚本。它是在我们的目标android设备的“/system/bin/monkey”,事实上这是一个android上面java程序启动的标准流程。
base=/system export CLASSPATH=$base/framework/monkey.jar trap "" HUP exec app_process $base/bin com.android.commands.monkey.Monkey $*
android中能够通过多种方式启动java应用,通过app_process命令启动就是当中一种。它能够帮忙注冊android JNI,而绕过dalvik以使用Native API(如我般不清楚的请百度)所做的主要事情例如以下:
- 设置monkey的CLASSPATH环境变量指向monkey.jar
- 通过app_process指定monkey的入口和传进来的全部參数启动上面CLASSPATH设定的monkey.jar
2.命令行參数解析
/** * Command-line entry point. * * @param args The command-line arguments */ public static void main(String[] args) { // Set the process name showing in "ps" or "top" Process.setArgV0("com.android.commands.monkey"); int resultCode = (new Monkey()).run(args); System.exit(resultCode); }
- 处理命令行參数
- 依据命令行參数启动不同的事件源,也就是我们的測试事件到底是从网络如monkeyrunner过来的还是monkey内部的random測试数据集过来的还是脚本过来的如此之类
- 跳入runMonkeyCyncle方法针对不同的事件源開始获取并运行不同的事件
private int run(String[] args) { ... if (!processOptions()) { return -1; } ...}
进去之后就是非常普通的读取命令行的參数然后一个个进行解析保存了,没有太多特别的东西,这里就直接贴出monkey的參数选项大家看看就好了:
3. 初始化測试事件源
private int run(String[] args) { ... if (mScriptFileNames != null && mScriptFileNames.size() == 1) { // script mode, ignore other options mEventSource = new MonkeySourceScript(mRandom, mScriptFileNames.get(0), mThrottle, mRandomizeThrottle, mProfileWaitTime, mDeviceSleepTime); mEventSource.setVerbose(mVerbose); mCountEvents = false; } else if (mScriptFileNames != null && mScriptFileNames.size() > 1) { if (mSetupFileName != null) { mEventSource = new MonkeySourceRandomScript(mSetupFileName, mScriptFileNames, mThrottle, mRandomizeThrottle, mRandom, mProfileWaitTime, mDeviceSleepTime, mRandomizeScript); mCount++; } else { mEventSource = new MonkeySourceRandomScript(mScriptFileNames, mThrottle, mRandomizeThrottle, mRandom, mProfileWaitTime, mDeviceSleepTime, mRandomizeScript); } mEventSource.setVerbose(mVerbose); mCountEvents = false; } else if (mServerPort != -1) { try { mEventSource = new MonkeySourceNetwork(mServerPort); } catch (IOException e) { System.out.println("Error binding to network socket."); return -5; } mCount = Integer.MAX_VALUE; } else { // random source by default if (mVerbose >= 2) { // check seeding performance System.out.println("// Seeded: " + mSeed); } mEventSource = new MonkeySourceRandom(mRandom, mMainApps, mThrottle, mRandomizeThrottle); mEventSource.setVerbose(mVerbose); // set any of the factors that has been set for (int i = 0; i < MonkeySourceRandom.FACTORZ_COUNT; i++) { if (mFactors[i] <= 0.0f) { ((MonkeySourceRandom) mEventSource).setFactors(i, mFactors[i]); } } // in random mode, we start with a random activity ((MonkeySourceRandom) mEventSource).generateActivity(); } ... mNetworkMonitor.start(); int crashedAtCycle = runMonkeyCycles(); mNetworkMonitor.stop(); ... }
事件源代表測试数据的事件是从哪里过来的。不同的event source会有不同的类来做对应的实现:
- MonkeySourceNetwork.java: 事件是从网络如monkeyrunner过来的,处理的是《MonkeyRunner源代码分析之与Android设备通讯方式》描写叙述的界面控制操作事件
- MonkeySourceNetworkVars.java: 事件也是从网络如monkeyrunner过来的,处理的是《MonkeyRunner源代码分析之与Android设备通讯方式》提到的getPropery事件
- MonkeySourceNetworkViews.java:事件也是从网络如monkeyrunner过来的。处理的是《MonkeyRunner源代码分析之与Android设备通讯方式》提到的Views相关的事件
- MonkeySourceRandom.java:事件是从monkey内部生成的随机事件集,也就是我们通过命令行启动monkey測试目标app的经常使用方式
- MonkeySourceRanodomeScript.java: 上面的随机内部数据源也能够通过指定setup脚本来创建
- MonkeySourceScript.java: 用户也能够遵循一定的规则编写monkey脚本来驱动monkey进行相关測试。与上面不同的是它不再是随机的
- 从指定的源获取命令
- 把命令翻译成monkey事件然后放到命令队列EventQueue
这些都是面向对象的面向接口编程的基础了。大家有不清楚的最好先去了解下java的一些基本知识。这样理解起来会快非常多。
4. 循环运行事件
private int run(String[] args) { ... int crashedAtCycle = runMonkeyCycles(); ... }
如前所述,runMonkeyCyles方法会依据不同的数据源開始一条条的获取事件并进行运行:
/** * Run mCount cycles and see if we hit any crashers. * <p> * TODO: Meta state on keys * * @return Returns the last cycle which executed. If the value == mCount, no * errors detected. */ private int runMonkeyCycles() { int eventCounter = 0; int cycleCounter = 0; boolean shouldReportAnrTraces = false; boolean shouldReportDumpsysMemInfo = false; boolean shouldAbort = false; boolean systemCrashed = false; // TO DO : The count should apply to each of the script file. while (!systemCrashed && cycleCounter < mCount) { ... MonkeyEvent ev = mEventSource.getNextEvent(); if (ev != null) { int injectCode = ev.injectEvent(mWm, mAm, mVerbose); ... } ... } .... }
注意这里的mEventSource就是我们上面提到的事件源的接口。它屏蔽了每一个事件实现类的详细细节,我们仅仅须要告诉这个接口我们如今须要取一条事件然后运行它。该结构依据面向对象的多态原理,就会自己主动取事件的实现类获得相应的事件进行返回。
本人曾经做过开发。所以还ok。
这里仅仅是做一个善意的提醒。
作者 |
自主博客 |
微信 |
CSDN |
天地会珠海分舵 |
|
服务号:TechGoGoGo 扫描码:
|
http://blog.csdn.net/zhubaitian |
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/116556.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...