1、MonkeyRunner:monkeyrunner工具提供了一个 API,使用此 API写出的程序可以在 Android代码之外控制 Android设备和模拟器。通过 monkeyrunner,您可以写出一个 Python程序去安装一个 Android应用程序或测试包,运行它,向它发送模拟击键,截取它的用户界面图片,并将截图存储于工作站上。monkeyrunner 工具的主要设计目的是用于测试功能/框架水平上的应用程序和设备,或用于运行单元测试套件,但您当然也可以将其用于其它目的。monkeyrunner工具与 用户界面/应用程序测试工具,也称为 monkey工具并无关联。monkey工具直接
2、运行在设备或模拟器的 adbshell中,生成用户或系统的伪随机事件流。而 monkeyrunner工具则是在工作站上通过 API定义的特定命令和事件控制设备或模拟器。1、MonkeyRunner 的特性1) 多设备控制:monkeyrunner API 可以跨多个设备或模拟器实施测试套件。您可以在同一时间接上所有的设备或一次启动全部模拟器(或统统一起) ,依据程序依次连接到每一个,然后运行一个或多个测试。您也可以用程序启动一个配置好的模拟器,运行一个或多个测试,然后关闭模拟器。 2) 功能测试:monkeyrunner 可以为一个应用自动贯彻一次功能测试。您提供按键或触摸事件的输入数值,然后
3、观察输出结果的截屏。 3) 回归测试:monkeyrunner 可以运行某个应用,并将其结果截屏与既定已知正确的结果截屏相比较,以此测试应用的稳定性。 4) 可扩展的自动化:由于 monkeyrunner是一个 API工具包,您可以基于 Python模块和程序开发一整套系统,以此来控制 Android设备。除了使用 monkeyrunner API之外,您还可以使用标准的 Python os和 subprocess模块来调用如 adb这样的 Android工具。 您还可以向 monkeyrunner API中添加您自己的类。我们将在 使用插件扩展 monkeyrunner一节中对此进行详细讨论
4、。2、一个简单的 monkeyrunner程序实例以下为一个简单的 monkeyrunner程序,它将会连接到一个设备,创建一个 MonkeyDevice对象。使用 MonkeyDevice对象,程序将安装一个 Android应用包,运行其中一个活动,并向其发送按键事件。程序接下来会将结果截图,创建一个 MonkeyImage对象,并使用这个对象截图将保存至.png 文件。# 导入此程序所需的 monkeyrunner模块 from com.android.monkeyrunner import MonkeyRunner, MonkeyDevice # 连接当前设备,返回一个 MonkeyDe
5、vice对象 device = MonkeyRunner.waitForConnection() # 安装 Android包,注意,此方法返回的返回值为 boolean,由此您可以判断安装过程是否正常 device.installPackage(myproject/bin/MyApplication.apk) # 运行此应用中的一个活动 device.startActivity(component=com.android.settings/.Settings) # 按下菜单按键 device.press(KEYCODE_MENU,DOWN_AND_UP) # 截取屏幕截图 result = d
6、evice.takeSnapShot# 将截图保存至文件 result.writeToFile(myproject/shot1.png,png)#获取指定区域的图像(200,400,200,400),注意两个括号result_static=result.getSubImage(200,400,200,400)#获取 d:shotbegin.png这张图片picture = MonkeyRunner.loadImageFromFile(d:shotbegin.png,png)#第二截图并获取相同的局部图像result_static2=picture.getSubImage(200,400,200
7、,400)#使用.sameAs()对比两张图片,并输出对比结果 True或 Falseend=result_static.sameAs(result_static2,1.0)print endmonkeyrunner执行测试时使用.takeSnapshot()截图,默认截取整个屏幕,包含了系统的状态栏。真实手机状态栏中包含如电量/信号量/消息提示等变量,使用.sameAs()对比整个屏幕的截图时就很容易出现错误。而使用.getSubImage()获得局部图像,然后再进行对比,就减少了 monkeyrunner执行结果出错的概率。result.getSubImage(200,400,200,40
8、0)中的指定区域值使用 Pixel Perfect获取坐标点,或者截图到本地后获取,先获取区域左上角和右下角坐标,前两个值是左上角左边,后两个值是右下角减左上角的坐标。3、The Monkeyrunner APImonkeyrunnerAPI于 com.android.monkeyrunner包中包含三个模块:1) MonkeyRunner:一个为 monkeyrunner程序提供工具方法的类。这个类提供了用于连接 monkeyrunner至设备或模拟器的方法。它还提供了用于创建一个 monkeyrunner程序的用户界面以及显示内置帮助的方法。 2) MonkeyDevice:表示一个设备或
9、模拟器。这个类提供了安装和卸载程序包、启动一个活动以及发送键盘或触摸事件到应用程序的方法。您也可以用这个类来运行测试包。 3) MonkeyImage :表示一个截图对象。这个类提供了截图、将位图转换成各种格式、比较两个 MonkeyImage对象以及写图像到文件的方法。 在 python程序中,您将以 Python模块的形式使用这些类。monkeyrunner 工具不会自动导入这些模块。您必须使用类似如下的 from语句:From com.android.monkeyrunner import MonkeyRunner,MonkeyDevice,MonkeyImage其中,为您想要导入的类名
10、。您可以在一个 from语句中导入超过一个模块,其间以逗号分隔。4、Monkeyrunner 命令语法4.1 导入需要的模块1 方式一:import sys,time,datetime,osfrom com.android.monkeyrunner import MonkeyRunner as mrfrom com.android.monkeyrunner import MonkeyDevice as mdfrom com.android.monkeyrunner import MonkeyImage as mi如果给导入的模块起了别名,就应该使用别名,而不能使用原名,否则会出现错误。比如连接
11、设备或模拟器,起了以上别名后,命令应该如下:device=mr.waitForConnection() 2 方式二:from com.android.monkeyrunner import MonkeyRunner,MonkeyDevice,MonkeyImage3 方式三:import com.android.monkeyrunner但是在使用时,就显得特别麻烦device=com.android.monkeyrunner.MonkeyRunner.waitForConnection() 4 方式四:我们也可以给它一个别名import com.android.monkeyrunner as
12、cam但是在使用时,就显得特别麻烦device=cam.MonkeyRunner.waitForConnection()#等待连接到设备,与模拟器连接,返回 monkeydevice 对象,代表连接的设备。没有报错的话说明连接成功。参数 1:超时时间,单位秒,浮点数。默认是无限期地等待。参数 2:串 deviceid,指定的设备名称。默认为当前设备(手机优先,比如手机通过 USB 线连接到 PC、其次为模拟器) 。默认连接:device=MonkeyRunner.waitForConnection()参数连接:device = mr.waitForConnection(1.0,emulator
13、-5554)4.2 向设备或模拟器安装要测试的 APKdevice.installPackage(myproject/bin/MyApplication.apk) #参数是相对或绝对 APK 路径路径级别用“/” ,不能用“” ,比如 d:wwwa.apk,而应该写成 d:/www/a.apk安装成功返回 true,此时查看模拟器我们可以在 IDLE 界面上看到安装的 APK 的图标了。4.3 从设备中删除指定的软件包,包括其相关的数据和调整缓存device.removePackage(myproject/bin/MyApplication.apk)删除成功返回 true。4.4 启动任意的
14、Activitydevice.startActivity(component=“ adb shell 命令,并返回结果,如果有的话device.shell(“.“)4.5 暂停目前正在运行的程序指定的秒数MonkeyRunner.sleep(秒数,浮点数)4.6 捕捉屏幕写入文件获取设备的屏蔽缓冲区,产生了整个显示器的屏蔽捕获。 (截图)result=device.takeSnapshot()返回一个 MonkeyImage 对象(点阵图包装) ,我们可以用以下命令将图保存到文件result.writeToFile(takeSnapshotresult1.png,png)MonkeyImage
15、.writeToFile(参数 1:输出文件名,也可以包括路径 ,参数 2:目标格式)写成功返回 true,否则返回 false4.7 type键盘上的类型指定的字符串,这相当于要求每个字符串中的字符按(键码,DOWN_AND_UP).字符串发送到键盘device.type(字符串 )4.8 唤醒设备屏幕(在设备屏幕上唤醒)device.wake()4.9 重新引导到指定的引导程序指定的设备device.reboot()5、Monkeyrunner 类的方法介绍 Alert(string message,string title,string okTitle)弹出一个对话框直到用户确认为止 m
16、essage:会话弹出的内容itle:会话标题,默认为 alertokTitle:会话确认按钮,默认为 ok Choice(string message,iterable choices,string title)显示一个对话框,选择你要添加那个 py 文件 message:显示在对话框中的提示信息。choices:一个迭代的包含一系列选择的 python 类型title:对话框的标题,默认为 input Help(string format)monkeyrunner 的一些帮助,这个和 api 差不多 Input(string message,string initialValue,stri
17、ng title,string okTiltle,string cancelTitle)用户可以在一个对话框里面输入内容用法:MonkeyRunner.input(message,initialValue,title,okTitle,cancelTitle)message:对话框显示的信息。The prompt message to display in the dialog.initialValue:提供给用户的初始化值,默认为空字符串。The initial value to supply to the user. The default is an empty string)title:
18、对话标题,默认为 input。The dialogs title. The default is InputokTitle:The text to use in the dialogs confirmation button. The default is OK.cancelTitle:The text to use in the dialogs cancel button. The default is Cancel.返回: The test entered by the user, or None if the user canceled the input;。 Sleep(float s
19、econds)让程序休息多长时间 WaitForConnection(float timeout,string deviceld)等待设备的连接用法:MonkeyRunner.waitForConnection(timeout,deviceId)timeout:等待的时间,默认为无限期deviceId:指定设备名称的一个规定表达式Returns: 一个表示已经连接的设备对象,A MonkeyDevice object6、MonkeyDevice 类的方法介绍 BroadcastIntent(string uri,string action,string data,string mimetype
20、,iterable categories dictionary extras,component component,iterable flages)对设备发送一个广播信号uri:信号的 uri Drag(tuple start,tuple end,float duration, integer steps)拖动屏幕,也就是划屏的一些操作用法:MonkeyDevice.drag(start,end,duration,steps)start:拖曳开始坐标 - The starting point for the drag (a tuple (x,y) in pixels)end:拖曳结束坐标点-
21、 The end point for the drag (a tuple (x,y) in pixelsduration:持续时间 - Duration of the drag in seconds (default is 1.0 seconds)steps:拖曳步骤 - The number of steps to take when interpolating points. (default is 10) GetProperty(string key)得到手机上的一些属性 GetSystemProperty(string key)得到一些系统属性 InstallPackage(strin
22、g path)将一个 apk 安装到手机里面 Instrument(string className.dictionary args)运行测试设备的指定包 Press(string name,dictionary type)按键(一些物理按键) Reboot(string into)重启手机 RemovePackage(string package)删除一些 apk Shell(string cmd)执行 adb shell 命令并返回结果 TakeSnapshot()截图 Touch(integer x,integer y,integer type)触摸 Type(string messag
23、e)输入一些字符串 Wake()唤醒手机点亮屏幕7、MonkeyImage 类的方法介绍 ConvertToBytes(string format)将图片转换为其他特殊的格式,将结果作为字符串返回,用这个方法将像素存取为特殊的格式,输出的字符串是一种更好的表现Format:目标格式,默认值为 png GetRawPixel(integer x,integer y)在 x,y 位置处获取一个单个的 ARGB 像素,参数 x,y 都是基于0 坐标,表示一个像素尺寸,x 向右增益,y 向下增益,得到一个图标的坐标,这个方法返回一个数组 GetRawPixelInt(integer x,integer
24、 y)得到一个图标的坐标,同上 .getRawPixel,只是返回的是一个整型 GetSubImage(tuple rect)复制一个图片的矩形区域rect: A tuple (x, y, w, h),x,y 指定矩形区域的左上角,w 为矩形宽,h 为矩形高 SameAs(MonkeyImage other,float percent)对比两张图片 WriteToFile(string path,string format)将得到的图片保存到一个文件夹里面用法:MonkeyImage.writeToFile(path,format)path:输入的文件名,可选择的包含路径format:目标格式
25、,默认为 png。return:如果输出成功返回 true。 GetHierarchyViewer() 获取设备的显示层次 StartActivity()在设备上开始一个活动8、MonkeyRunner 常用的按键介绍 Home 键:KEYCOD_HOMEBack 键: KEYCODE_BACKsend 键:KEYCODE_CALLend 键:KEYCODE_ENDCALL上导航键:KEYCODE_DPAD_UP(现在手机已经没有这个键)下导航键:KEYCODE_DPAD_DOWN(现在手机已经没有这个键)左导航:KEYCODE_DPAD_LEFT 现在手机已经没有这个键右导航键:KEYCOD
26、E_DPAD_RIGHT 现在手机已经没有这个键ok 键:KEYCODE_DPAD_CENTER上音量键:KEYCODE_VOLUME_UP 下音量键:KEYCODE_VOLUME_DOWNpower 键: KEYCODE_POWERcamera 键:KEYCODE_CAMERAmenu 键:KEYCODE_MENUsearch 键:KEYCODE_SEARCHcall 键:KEYCODE_CALL按下 HOME 键 device.press(KEYCODE_HOME,MonkeyDevice.DOWN_AND_UP) 按下 BACK 键 device.press(KEYCODE_BACK,M
27、onkeyDevice.DOWN_AND_UP) 按下下导航键 device.press(KEYCODE_DPAD_DOWN,MonkeyDevice.DOWN_AND_UP) 按下上导航键 device.press(KEYCODE_DPAD_UP,MonkeyDevice.DOWN_AND_UP) 按下 OK 键 device.press(KEYCODE_DPAD_CENTER,MonkeyDevice.DOWN_AND_UP)device.press(KEYCODE_ENTER,MonkeyDevice.DOWN_AND_UP)#输入回车device.press(KEYCODE_BACK,
28、MonkeyDevice.DOWN_AND_UP)#点击返回Monkeyrunner 脚本的例子: 添加联系人脚本:import sys,time,datetime,osfrom com.android.monkeyrunner import MonkeyRunner,MonkeyDevice,MonkeyImagedevice=MonkeyRunner.waitForConnection()MonkeyRunner.sleep(1)device.startActivity(component=com.android.contacts/.activities.PeopleActivity)Mo
29、nkeyRunner.sleep(1)device.touch(654,1237,DOWN_AND_UP)MonkeyRunner.sleep(1)device.type(“yidong“)MonkeyRunner.sleep(1)device.press(KEYCODE_BACK, MonkeyDevice.DOWN_AND_UP)MonkeyRunner.sleep(1)device.touch(293,790,DOWN_AND_UP)MonkeyRunner.sleep(1)device.type(“10086“)MonkeyRunner.sleep(1)device.touch(657
30、,94,DOWN_AND_UP)MonkeyRunner.sleep(1)result = device.takeSnapshot()result.writeToFile(“d:people.png“, “png“)hui=result.getSubImage(200,400,200,400)picture = MonkeyRunner.loadImageFromFile(d:people.png,png)hui1=picture.getSubImage(200,400,200,400)ggg=hui.sameAs(hui1,1.0)print ggg 打电话脚本(用 id 判断)注:用 id
31、 判断速度比较慢import sys,time,datetime,osfrom com.android.monkeyrunner import MonkeyRunner,MonkeyDevice,MonkeyImagefrom com.android.monkeyrunner.easy import EasyMonkeyDevice,Bydevice=MonkeyRunner.waitForConnection()MonkeyRunner.sleep(2)#device.press(KEYCODE_HOME, MonkeyDevice.DOWN_AND_UP)#MonkeyRunner.sle
32、ep(2)device.startActivity(component=com.android.contacts/.activities.DialtactsActivity) MonkeyRunner.sleep(2)device.touch(By.id(id/one,MonkeyDevice.DOWN_AND_UP)MonkeyRunner.sleep(1)device.touch(By.id(id/zero),DOWN_AND_UP)MonkeyRunner.sleep(1)device.touch(By.id(id/zero),DOWN_AND_UP)MonkeyRunner.sleep
33、(1)device.touch(By.id(id/eight),DOWN_AND_UP)MonkeyRunner.sleep(1)device.touch(By.id(id/six),DOWN_AND_UP)MonkeyRunner.sleep(1)device.touch(By.id(id/dialButtonGreen),DOWN_AND_UP)MonkeyRunner.sleep(10)#device.touch(By.id(id/one),DOWN_AND_UP)result = device.takeSnapshot()hui=result.getSubImage(200,400,200,400)#result.writeToFile(“d:shotbegin.png“, “png“)picture = MonkeyRunner.loadImageFromFile(d:shotbegin.png,png)hui1=picture.getSubImage(200,400,200,400)ggg=hui.sameAs(hui1,1.0)print gggMonkeyRunner.sleep(2)device.press(KEYCODE_BACK, MonkeyDevice.DOWN_AND_UP)#device.reboot()MonkeyRunner.sleep(1)