Android 程序员必须掌握的三种自动化测试方法

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

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

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

1.Monkey

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

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

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

举例来说:

1 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里最常用的登录功能,我们就可以编写一个脚本来完成。

 1 # coding=UTF-8
 2 from com.android.monkeyrunner import MonkeyRunner as mr
 3 from com.android.monkeyrunner import MonkeyDevice as md
 4 from com.android.monkeyrunner import MonkeyImage as mi
 5 from com.android.monkeyrunner.easy import EasyMonkeyDevice
 6 from com.android.monkeyrunner.easy import By
 7
 8 #定义安装文件路径
 9 installPackage = ‘C:\\Users\\Administrator\\ViewPageTest\\app\\build\\outputs\\apk\\app-debug.apk‘
10
11 #要测试的程序的包名
12 apkPackageName =‘com.example.administrator.viewpagetest‘
13
14 #要启动的第一个activity的名称
15 initActivityName=apkPackageName+"/com.example.administrator.viewpagetest.MainActivity"
16
17
18 device = mr.waitForConnection()
19
20
21 #安装apk包
22 device.installPackage(installPackage.decode(‘utf-8‘))
23
24
25 #启动应用程序
26 device.startActivity(component=initActivityName)
27 #防止启动首页面 需要时间过长
28 mr.sleep(3)
29
30 easy_device = EasyMonkeyDevice(device)
31
32 mr.sleep(3)
33
34
35
36 easy_device.type(By.id(‘id/username_et‘),‘zhangsan‘)
37 # 这里的mr静止 主要用于演示demo上的gif效果
38 mr.sleep(2)
39 easy_device.type(By.id(‘id/password_et‘),‘123456‘)
40 mr.sleep(2)
41 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里的改动:

 1 apply plugin: ‘com.android.application‘
 2
 3 android {
 4     compileSdkVersion 23
 5     buildToolsVersion "23.0.2"
 6
 7     defaultConfig {
 8         //不要遗漏这句话
 9         testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
10         applicationId "com.example.administrator.testcaseone"
11         minSdkVersion 18
12         targetSdkVersion 23
13         versionCode 1
14         versionName "1.0"
15     }
16     buildTypes {
17         release {
18             minifyEnabled false
19             proguardFiles getDefaultProguardFile(‘proguard-android.txt‘), ‘proguard-rules.pro‘
20         }
21     }
22 }
23
24 dependencies {
25     compile fileTree(dir: ‘libs‘, include: [‘*.jar‘])
26     testCompile ‘junit:junit:4.12‘
27     //这个地方要注意了 studio自带的里面版本号一般都比较高,如果出错的话 你要手动把这个版本号调低一点
28     compile ‘com.android.support:appcompat-v7:23.0.1‘
29     compile ‘com.android.support:design:23.0.1‘
30     //对这个androidTestCompile不理解的 可以参考我的博客里讲gradle的那篇
31     androidTestCompile ‘com.android.support.test:runner:0.4‘
32     androidTestCompile ‘com.android.support.test:rules:0.4‘
33     androidTestCompile ‘com.android.support.test.uiautomator:uiautomator-v18:2.1.2‘
34 }

然后变更一个conifg

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

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

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

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

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

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

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

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

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

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

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

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

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

时间: 2025-01-02 18:54:17

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

给Android程序员的一些面试建议

前言 应大家的邀请,写一篇关于Android面试相关的博客,需要说明的是本文只针对Android应用开发,不针对rom开发以及逆向工程.我想面试对于程序员来说是很重要的一件事件,面试结果的好坏直接决定了能否进入某个公司以及以什么级别和待遇进入某个公司.我参加面试的经验并不多,但是以面试官的身份面试别人倒是有很多次,所以我可以结合这些经验来介绍下如何更好地把握一个面试. 什么是合适的候选者 在介绍如何面试之前,这里先从公司的角度来分析:"到底什么样的候选者是公司所需要的技术人才?"就我在

程序员赚钱致富的6种方法

我认识一个朋友,也是程序员出身,他在一家还不错的外企上班,每个月工资收入也就差不多 15K,五年的工作经验了,在他面前,我算是小弟.那天我们几个朋友一起打完球就去附近的饭馆吃饭,环境还不错,于是就边吃边聊工作.赚钱的事情. 那天了解到,他不仅拿着 15K 的高薪,业余还有着更高的收入,从聊天中,我总结了几点程序员赚钱的技巧,分享给大家,也许你可以参考一下,哪天发财了记得回到这篇文章中来赞一下. 一.Google Adsense 利用 Adsense 可以将广告发布到你的网站上去,通过访客点击广告

据说年薪30万的Android程序员必须知道的帖子

据说年薪30万的Android程序员必须知道的帖子 标签: android 2015-03-12 16:52 10443人阅读 评论(10) 收藏 举报 Android中国开发精英 目前包括: Android开源项目第一篇——个性化控件(View)篇       包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.ProgressBar.TextView.ScrollView.TimeView.TipView.FlipVi

【转】LayoutInflater——80%的Android程序员对它并不了解甚至错误使用

这个标题起的有点夸张哈,但是LayoutInflater这个类的一些用法,在Android开发者使用的过程中,确实存在着一些很普遍的误区,最起码我研究的这么多小项目的源代码,基本上都在错误的使用这个类.今天,看到了一篇文章讲LayoutInflater的用法,瞬间感觉自己对这个类确实不够了解,于是简单的看了下LayoutInflater类的源代码,对这个类有了新的认识. 首先,LayoutInflater这个类是用来干嘛的呢? 我们最常用的便是LayoutInflater的inflate方法,这

LayoutInflater——80%的Android程序员对它并不了解甚至错误使用

这个标题起的有点夸张哈,但是LayoutInflater这个类的一些用法,在Android开发者使用的过程中,确实存在着一些很普遍的误区,最起码我研究的这么多小项目的源代码,基本上都在错误的使用这个类.今天,看到了一篇文章讲LayoutInflater的用法,瞬间感觉自己对这个类确实不够了解,于是简单的看了下LayoutInflater类的源代码,对这个类有了新的认识. 首先,LayoutInflater这个类是用来干嘛的呢? 我们最常用的便是LayoutInflater的inflate方法,这

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

每个Android 程序员都不是Android应用开发之路上孤军奋战的一个人,GitHub上浩如烟海的开源框架或类库就是前人为我们发明的轮子,有的轮子能提高软件性能,而有的轮子似乎是以牺牲性能为代价换取编程速度.擅长利用轮子的程序员已经遥遥领先,不擅长利用轮子的程序员总是嫌前人发明的轮子不够圆,自己造个方轮子上路后才发现落后了. 作者:玖哥来源:51CTO|2017-10-19 16:06 移动端 收藏 分享 [51CTO.com原创稿件]每个Android 程序员都不是Android应用开发之

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

1.前言 网络通信一直是Android项目里比较重要的一个模块,Android开源项目上出现过很多优秀的网络框架,从一开始只是一些对HttpClient和HttpUrlConnection简易封装使用的工具类,到后来Google开源的比较完善丰富的Volley,再到如今比较流行的Okhttp.Retrofit. 要想理解他们之间存在的异同(或者具体点说,要想更深入地掌握Android开发中的网络通信技术),必须对网络基础知识.Android网络框架的基本原理等做到心中有数.信手拈来,关键时刻才能

@Android程序员,请掌握这些核心生存技能!

大佬你好,作为一个Android开发者,在学习过程中有什么需要注意的? 这个问题其实也不太好回答,因为要注意的问题蛮多的,多得就像雾霾中的颗粒.不过,有趣的是,流年似水这两次的提问中,对我的称呼大有不同. 再说,我也不是什么"大佬",无非比较接地气一点,喜欢和大家交流分享的感觉而已. 那么接下来,我就针对这个提问,做一些回答.希望对有类似问题的同学起到一个参考的作用.将以以下几个方面做出回答: 1.不断学习2.分享一下我的学习方法3.移动开发者核心生存技能(高能!) 为什么要不断学习

给年后面试的Android程序员的一些面试建议

前言 今天写一篇关于Android面试相关的博客,需要说明的是本文只针对Android应用开发,不针对rom开发以及逆向工程.我想面试对于程序员来说是很重要的一件事件,面试结果的好坏直接决定了能否进入某个公司以及以什么级别和待遇进入某个公司.我参加面试的经验并不多,但是以面试官的身份面试别人倒是有很多次,所以我可以结合这些经验来介绍下如何更好地把握一个面试. 什么是合适的候选者 在介绍如何面试之前,这里先从公司的角度来分析:"到底什么样的候选者是公司所需要的技术人才?"就我在百度的一些