在日常的开发中,尤其是app开发,因为不像web端那样 出错以后可以热更新,所以app开发 一般对软件质量有更高的要求(你可以想一下 一个发出去的版本如果有重大缺陷 需要强制更新新客户端是多么蛋疼的事情)。

恩,所以我们app的开发者 一定要学会自己测试自己的代码 自己测试自己的app,不要寄希望于测试来帮你找bug,实际上,我工作多年的经验告诉,绝大多数隐藏极深的bug 都是开发自己发现的。

所以 今天就来教大家几招,如何测试自己的app,测试自己的模块。

1.Monkey

http://developer.android.com/intl/zh-cn/tools/help/monkey.html

这个工具是最简单的,我主要用他来压力测试,所谓压力测试就是 乱点。。。模拟各种各样奇怪的操作 看你的app能不能抗的住。

可以简单看一下 这个命令的用法。看一下help 介绍的参数说明。

举例来说:

 android shell monkey -p 你想测试程序的包名 -v 500

比如 我现在想看看android 系统自带的日历应用 在压力下表现如何。

你看 这个地方模拟器自己就开始疯狂点击了。当然在实际使用中,我们都是会把次数调到 几十万次到几百万次,然后下班以后开始跑,第二天来看结果 看看在哪里出了问题~~。基本上

每日构建完毕以后都会跑一下。Monkey基本上 就是这样使用的。非常简单 但是作用也非常有限。不过可以极大帮助你 找出你app的一些隐藏极深的bug。

比如evernote,这个我平常使用的软件 我自己是没有碰到过crush的,但是你跑一下monkey,1个多小时 就崩溃了。。。。。所以monkey是提升软件质量的 好帮手。

2.MonkeyRunner

http://developer.android.com/intl/zh-cn/tools/help/monkeyrunner_concepts.html

这个相对于Monkey 来说 就是真正意义上的 自动化测试工具了。只需要编写脚本即可完成 我们平时所需要的 大部分  冒烟用例等等。

尤其是在4.x以下的机型里,由于无法使用uiautomator, MonkeyRunner几乎就是唯一的自动化测试编写办法。

下面我举个例子,比如我们app里最常用的登录功能,我们就可以编写一个脚本来完成。

 # coding=UTF-8
from com.android.monkeyrunner import MonkeyRunner as mr
from com.android.monkeyrunner import MonkeyDevice as md
from com.android.monkeyrunner import MonkeyImage as mi
from com.android.monkeyrunner.easy import EasyMonkeyDevice
from com.android.monkeyrunner.easy import By #定义安装文件路径
installPackage = 'C:\\Users\\Administrator\\ViewPageTest\\app\\build\\outputs\\apk\\app-debug.apk' #要测试的程序的包名
apkPackageName ='com.example.administrator.viewpagetest' #要启动的第一个activity的名称
initActivityName=apkPackageName+"/com.example.administrator.viewpagetest.MainActivity" device = mr.waitForConnection() #安装apk包
device.installPackage(installPackage.decode('utf-8')) #启动应用程序
device.startActivity(component=initActivityName)
#防止启动首页面 需要时间过长
mr.sleep(3) easy_device = EasyMonkeyDevice(device) mr.sleep(3) easy_device.type(By.id('id/username_et'),'zhangsan')
# 这里的mr静止 主要用于演示demo上的gif效果
mr.sleep(2)
easy_device.type(By.id('id/password_et'),'')
mr.sleep(2)
easy_device.touch(By.id('id/submit_bt'),md.DOWN_AND_UP)

然后运行他 看看效果:

你看上面的脚本 完成了 自己安装apk 输入用户名和密码 并且点击登录按钮的过程。

有人问,你这个模拟登录的过程是模拟出来了,那我怎么知道 到底登录成功没有呢?

其实也很简单。主要有几个方法。

1.登录成功以后你这个页面肯定是要跳转到主界面的对吧,你就用脚本执行下shell命令 看看主页面 是否在栈的最上方?(前面我的activity 启动模式那篇博客里讲过这个命令的)

2.你可以在log 里面 打印出登录成功这个消息 然后用脚本捕捉到这个log 日志 就知道是否登录成功了。

3.甚至你还可以捕获界面上某个控件的文字值。

4.比较某个操作结束后的 截屏。保存为图像以后 和正确操作以后的图像进行对比。

这里我就不继续往下写这个脚本了。有兴趣的同学可以自己尝试 完成日常工作里的 那些冒烟用例。(意义重大,否则每次发版本 你的那些用例全部要用手点击手机完成一遍 那多麻烦!)

此外 我们还可以利用recorder来录制脚本,然后再反过来用python执行这个脚本 来执行我们的测试过程。这个方法 我就不做详细分析了,很简单。(但是要注意 这个方法 启动的 捕捉器 在多数情况下都非常卡顿,所以采用率不高。)

3.UiAutomator

http://developer.android.com/intl/zh-cn/tools/testing-support-library/index.html

这个工具我个人认为是所有android 程序员都必须要掌握的,有了这个强大的工具,我们就可以负责任的对自己的代码 说 木问题,ok!

此工具 能模拟几乎所有对android设备的操作。

而且代码也非常简单 全部都是java代码,并且android的api 他还几乎都能够使用。简直酷到没有朋友!比android studio 自带的ApplicationTestCase 强到不知道哪里去了。

在这之前 你需要对gradle脚本有少许了解。具体可参见我的blog http://www.cnblogs.com/punkisnotdead/p/5029125.html

这个工具的原理实际上和http://www.cnblogs.com/punkisnotdead/p/4885572.html 里面提到的辅助服务是差不多的。都是利用的那个service。

你只要会写UiAutomator testcase,就意味着你的代码 几乎是永远不会出错哒~~

好,下面给出一个基本的例子 来让你明白 为何这个工具这么吊。

首先 给出gradle里的改动:

 apply plugin: 'com.android.application'

 android {
compileSdkVersion 23
buildToolsVersion "23.0.2" defaultConfig {
//不要遗漏这句话
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
applicationId "com.example.administrator.testcaseone"
minSdkVersion 18
targetSdkVersion 23
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
} dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
testCompile 'junit:junit:4.12'
//这个地方要注意了 studio自带的里面版本号一般都比较高,如果出错的话 你要手动把这个版本号调低一点
compile 'com.android.support:appcompat-v7:23.0.1'
compile 'com.android.support:design:23.0.1'
//对这个androidTestCompile不理解的 可以参考我的博客里讲gradle的那篇
androidTestCompile 'com.android.support.test:runner:0.4'
androidTestCompile 'com.android.support.test:rules:0.4'
androidTestCompile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
}

然后变更一个conifg

然后我们就可以在studio里面直接run我们的testcase啦:

好,然后我们来假设一个场景,假设我们现在要做的功能是 有一个界面,界面上有2个输入框,你在这2个输入框里输入数字以后,点击计算按钮

另外一个textview 就会显示出来 2个数字相加的结果。 那我们的testcase就要来完成 用户模拟操作的这个过程 并且看看结果是否和我们预想中的相匹配!

(假设我们现在的android代码里这个计算的代码是有错误的)

                 textView.setText(Integer.parseInt(et1.getText().toString()) + Integer.parseInt(et2.getText().toString()) + 1 + "");

你看 这里 我故意加了一个1.

然后写我们的testcase 注意testcase的位置

然后看一下这个testcase的代码:

 package com.example.administrator.testcaseone;

 import android.support.test.uiautomator.UiAutomatorInstrumentationTestRunner;
import android.support.test.uiautomator.UiAutomatorTestCase;
import android.support.test.uiautomator.UiDevice;
import android.support.test.uiautomator.UiObject;
import android.support.test.uiautomator.UiObjectNotFoundException;
import android.support.test.uiautomator.UiScrollable;
import android.support.test.uiautomator.UiSelector; /**
* Created by Administrator on 2016/1/5.
*/
public class FirstUiautomatorTest extends UiAutomatorTestCase { public void testDemo() throws UiObjectNotFoundException { UiDevice.getInstance(getInstrumentation());
//19-27 行 其实就是用这个框架提供的功能来直接启动你的app.
//这里其实主要就是要找到你的app那个textview 然后点击他 具体api自己去慢慢看吧
getUiDevice().pressHome();
UiScrollable appViews = new UiScrollable(new UiSelector()
.scrollable(true));
UiObject myApp = appViews.getChildByText(new UiSelector()
.className("android.widget.TextView"), "TestCaseOne");
//要等到新的窗口出来才继续往下走
myApp.clickAndWaitForNewWindow();
//29-32行 就很简单了,无非就是找到界面上的元素。
UiObject et1 = new UiObject(new UiSelector().resourceId("com.example.administrator.testcaseone:id/et"));
UiObject et2 = new UiObject(new UiSelector().resourceId("com.example.administrator.testcaseone:id/et2"));
UiObject bt1 = new UiObject(new UiSelector().resourceId("com.example.administrator.testcaseone:id/bt1"));
UiObject tv1 = new UiObject(new UiSelector().resourceId("com.example.administrator.testcaseone:id/tv1"));
//这里的sleep只是为了gif动画能显示的更清楚罢了,一般我们自己写的时候为了用例速度快一点 是不会加sleep的
//某些特殊场景除外
et1.setText("12");
sleep(3000);
et2.setText("21");
sleep(3000);
bt1.click();
//12和21相加 明显应该是33,所以判断下 我们的代码是否正确
assertEquals(33, Integer.parseInt(tv1.getText().toString())); }
}

然后直接run我们的这个defaluttest,看看模拟器会发生什么?

自动都帮你模拟了用户的操作,这个testcase就跑完了,然后看下我们的studio:

显然的也报错了。并且这个工具能自动捕获ui错误哦~~什么anr 之类的 都不在话下。有了他,我们就能为自己写的代码负责了,每次发版本之前 跑跑我们写好的testcase

基本上就能保证我们app的绝大多数流程是ok的~~所以这个工具一定要掌握!至于他其他好多api 我就不过多介绍了,留给你们自己去探索吧!

Android 程序员必须掌握的三种自动化测试方法的更多相关文章

  1. Android程序员必知必会的网络通信传输层协议——UDP和TCP

    1.点评 互联网发展至今已经高度发达,而对于互联网应用(尤其即时通讯技术这一块)的开发者来说,网络编程是基础中的基础,只有更好地理解相关基础知识,对于应用层的开发才能做到游刃有余. 对于Android ...

  2. 迈向高阶:优秀Android程序员必知必会的网络基础

    1.前言 网络通信一直是Android项目里比较重要的一个模块,Android开源项目上出现过很多优秀的网络框架,从一开始只是一些对HttpClient和HttpUrlConnection简易封装使用 ...

  3. IT观察】网络通信、图片显示、数据库操作……Android程序员如何利用开源框架

    每个Android 程序员都不是Android应用开发之路上孤军奋战的一个人,GitHub上浩如烟海的开源框架或类库就是前人为我们发明的轮子,有的轮子能提高软件性能,而有的轮子似乎是以牺牲性能为代价换 ...

  4. Android 程序员必须知道的 53 个知识点

    1. android 单实例运行方法 我们都知道 Android 平台没有任务管理器,而内部 App 维护者一个 Activity history stack 来实现窗口显示和销毁,对于常规从快捷方式 ...

  5. (转载) 据说年薪30万的Android程序员必须知道的

    据说年薪30万的Android程序员必须知道的帖子 标签: android 2015-03-12 16:52 28705人阅读 评论(14) 收藏 举报 Android中国开发精英 目前包括: And ...

  6. Android程序员问答题

    前言 最近三个月内,不断地进行移动应用开发在线测试题,也积累了不一样的知识.这也将对android studio有很好的掌握,对将来面试也很有好处.那么我就分享给大家.分享是一种幸福,这是一种质的飞越 ...

  7. 面试利器!字节跳动2021年Android程序员面试指导小册已开源

    整份手册分为两个部分,分别是:Java部分.Android部分.数据结构与算法篇.字节跳动2020年全年面试题总结篇! 每个知识点都有左侧导航书签页,看的时候十分方便,由于内容较多,这里就截取一部分图 ...

  8. 【定有惊喜】android程序员如何做自己的API接口?php与android的良好交互(附环境搭建),让前端数据动起来~

    一.写在前面 web开发有前端和后端之分,其实android还是有前端和后端之分.android开发就相当于手机app的前端,一般都是php+android或者jsp+android开发.androi ...

  9. Android开发中完全退出程序的三种方法

    参考: http://android.tgbus.com/Android/tutorial/201108/363511.shtml Android程序有很多Activity,比如说主窗口A,调用了子窗 ...

随机推荐

  1. Linux多线程之同步2 —— 生产者消费者模型

    思路 生产者和消费者(互斥与同步).资源用队列模拟(要上锁,一个时间只能有一个线程操作队列). m个生产者.拿到锁,且产品不满,才能生产.当产品满,则等待,等待消费者唤醒.当产品由空到不空,通知消费者 ...

  2. java反射机制浅谈

    一.Java的反射机制浅谈 最近研究java研究得很给力,主要以看博文为学习方式.以下是我对java的反射机制所产生的一些感悟,希望各位童鞋看到失误之处不吝指出.受到各位指教之处,如若让小生好好感动, ...

  3. vim中如何引用自定义模板文件

    我们在使用vim新建文件时可以引用自定义模板,来避免重复的数据格式处理花费太多时间. 实现方法很简单,只需要2步即可:1. 在.vim/template目录放入自己的模板文件(如shellconfig ...

  4. C#中out的用法

    out的用法 out 关键字会导致参数通过引用来传递.这与 ref 关键字类似,不同之处在于 ref 要求变量必须在传递之前进行初始化.若要使用 out 参数,方法定义和调用方法都必须显式使用 out ...

  5. 今天来做一个PHP电影小爬虫。

    今天来做一个PHP电影小爬虫.我们来利用simple_html_dom的采集数据实例,这是一个PHP的库,上手很容易.simple_html_dom 可以很好的帮助我们利用php解析html文档.通过 ...

  6. Java:内部类

    1.内部类的定义: 一个内部类可以定义在另一个类里,可以定义在函数里,甚至可以作为一个表达式的一部分. 2.内部类的分类: Java中的内部类共分为四种: 成员内部类member inner clas ...

  7. Java:日历类、日期类、数学类、运行时类、随机类、系统类

    一:Calendar类 java.util 抽象类Calendar   1.static Calendar getInstance()使用默认时区和语言环境获得一个日历. 2. int get(int ...

  8. Android TabHost中Activity之间传递数据

    例子1: TabHost tabhost = (TabHost) findViewById(android.R.id.tabhost); tabhost.setup(this.getLocalActi ...

  9. Data Base sqlServer基础知识

    sqlServer   基础知识 大纲 创建数据库 1 创建表 2 备份表 3 删除表 4 修改表 5 查询出重复的数据 6 增删改查 7 添加约束 8 分页存储过程 9 排序 10 类型转换 11 ...

  10. AngularJs-ui modal 传参数

    最近开始学习 AnjularJs: 看了两天项目的代码开始动手完成项目中的功能,碰到些问题记录下备忘:方便以后再碰到这样疑惑的coder. 参见 Angular-ui  modal 传递 header ...