Android技术博客

Android 单元测试

平时单元测试写得比较少,以至于现在都有点陌生了,现在简单做下整理,记录下来方便日后复习。

使用Android Studio进行单元测试

android studio的单元测试由于只是模拟android开发环境,但是其不是真正的android开发环境,所以不能测试UI功能,不能测试需要硬件支持的功能(比如蓝牙,wifi等),不能测试App跳转等等,那么其可以测试那些内容呢?

  • 测试一些数据性的功能,比如加载网络数据
  • 测试SharedPerferences,测试数据库,测试函数等
  • 工具类的测试,比如验证时间,转化格式,正则验证等等

在Android Studio中进行单元测试并不需要什么插件或者过多的配置,Android Studio本身就集成了测试环境,无论是单纯的java代码单元测试还是依赖Android SDK的Android代码单元测试都可以。


Android单元测试框架主要有:JUnit和Robolectric,本文就不详细介绍。
1. 使用JUnit进行单元测试
参考:http://www.open-open.com/lib/view/open1328152424546.html
2. Robolectric
http://robolectric.org/getting-started/
在安卓模拟器或者真机上跑测试用例速度很慢。构建、部署、启动app,通常需要花费很久。Robolectric提供一种更好的方式,它模拟了android sdk中的jar包,可以直接在jvm中运行测试用例,这样就大大节省了时间。
另外,使用JUnit对fragment进行单元测试,可以参考该文章:http://blog.denevell.org/android-testing-fragments.html


首先我们需要明确安卓的单元测试主要分为两种类型:

  • 在开发主机Java虚拟机上运行的Junit Test
  • 在Android真机或者虚拟机上运行的Instrumented unit tests

因为Junit Test免去了将apk包安装到android设备上的步骤,因此速度上会比Instrumented unit tests快很多。但是跟android相关的一些库只能运行在android设备上,而无法再本机Java虚拟机上运行,比如很多包名中带android的库。因此我们这里需要做一个选择,如果某些类没有用到android相关的库,则完全可以使用Junit Test,加快测试速率。


我们新创建一个工程后,项目在Android Studio的Project中应该是这样的,其中androidTest包中存放的是Instrumented unit tests的文件,而test中则是Junit Test,这两个文件夹在项目创建的时候AS都会帮我们创建好:
QQ截图20170118141231.png

分析一下:
(1)androidTest这个文件夹里的测试类主要是对android的例子进行单元测试。需要运行于设备之上。
(2)test这个文件夹的测试类主要对java的例子进行单元测试,无需运行于设备之上。

在你的gradle中加入Junit的依赖,注意这里的依赖方式是测试期间的依赖(testCompile)

1
testCompile 'junit:junit:4.12'

编写java测试用例

如果所写的测试代码没有使用android sdk(android.*下的代码),那么可以在test目录下新建,本例中即为ExampleUnitTes

注意测试用例即一个public void的方法,并且加上@Test注解,这是Junit的标准用法

1
2
3
4
5
6
public class ExampleUnitTest {
@Test
public void addition_isCorrect() throws Exception {
assertEquals(4, 2 + 2);
}
}

编写好测试代码后,有如下两种方法运行

QQ截图20170118141829.png

成功的例子
QQ截图20170118141733.png

失败的例子

QQ截图20170118162359.png

还有一种失败的情况 ,因为用了Android的代码,所以这样子了,这就引出我们的Android测试例子。
QQ截图20170118162444.png

编写Android测试用例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**
* Created by dell on 2017/1/18.
*
* 测试用例类需要使用注解:@MediumTest@RunWith(AndroidJUnit4.class)
*/
@MediumTest
@RunWith(AndroidJUnit4.class)
public class MTest {
/**
*
* 我们在编写测试函数的时候需要注意两点:
测试函数需要为public
测试函数需要添加@Test注解
*/
@Test
public void test1() {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getTargetContext();
assertEquals("uuch.com.android_activityanim", appContext.getPackageName());
Log.i("tag", "$");
assertEquals("result:", 123, 100 + 33);
}
}

运行方式和以上 java测试用例一样。

常见的单元测试框架

JUnit4

这是Java界用的最广泛,也是最基础的一个框架 。在Android项目里面使用JUnit是很简单的,你只需要将JUnit这个library加到你的dependencies里面。

1
testCompile 'junit:junit:4.12'

如果你通过AndroidStudio创建一个项目,这个dependency默认是加上了的,所以你甚至这步都可以省略。

上面java测试用例就是用得这个框架,相对来说是比较简单,也是比较容易理解的 。其中Assert部分,可以帮我们验证一个方法的返回结果。然而,这些只能帮我们测试有返回值的那些方法;没有返回值的方法,即void方法,则要通过另外一个框架,Mockito,来验证它的正确性

Mockito

这个是Java界使用最广泛的一个mock框架。

Robolectric3.0

官网地址
Github地址
Robolectric环境搭建(Android Studio + Gradle)
Android 单元测试–Robolectric

Instrumentation与Roboletric都是针对 Android 进行单元测试的框架,前者在执行 case 时候是以 Android JUnit 的方式运行,因此必须在真实的 Android 环境中运行(模拟器或者真机),而后者则是以 Java Junit 的方式运行,这里就脱离了对 Android 环境的依赖,而可以直接将 case 在 JVM 中运行,大赞,因此很适合将 Roboletric 用于 Android 的测试驱动开发。


官网介绍
Running tests on an Android emulator or device is slow! Building, deploying, and launching the app often takes a minute or more. That’s no way to do TDD. There must be a better way.
Wouldn’t it be nice to run your Android tests directly from inside your IDE? Perhaps you’ve tried, and been thwarted by the dreaded java.lang.RuntimeException: Stub!?
Robolectric is a unit test framework that de-fangs the Android SDK jar so you can test-drive the development of your Android app. Tests run inside the JVM on your workstation in seconds. With Robolectric you can write tests like this:

对于Android app来说,写起单元测试来瞻前顾后,一方面单元测试需要运行在模拟器上或者真机上,麻烦而且缓慢,另一方面,一些依赖Android SDK的对象(如Activity,TextView等)的测试非常头疼,Robolectric可以解决此类问题,它的设计思路便是通过实现一套JVM能运行的Android代码,从而做到脱离Android环境进行测试。

使用
在app/gradle中加入依赖关系

1
2
3
dependencies {
testCompile "org.robolectric:robolectric:3.2.2"
}

在app/src/test 目录下,创建测试用例 ,运行方式和上面的一样, 第一次会比较慢

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@RunWith(RobolectricTestRunner.class)
@Config(constants = BuildConfig.class,sdk = 19)
public class JustTest {
@Before
public void setUp() throws URISyntaxException {
//输出日志
ShadowLog.stream = System.out;
}
@Test
public void testActivity() {
MainActivity sampleActivity = Robolectric.setupActivity(MainActivity.class);//创建Activity
Log.i("","_____________" + sampleActivity.getClass().getSimpleName());
Assert.assertNotNull(sampleActivity);//判断是否为空
Assert.assertEquals(sampleActivity.getClass().getSimpleName(), "MainActivity");//获取Activity的标题
}
}

如果想要支持Fragment

1
2
// 如果使用support的Fragment,需添加以下依赖
testCompile 'org.robolectric:shadows-support-v4:3.1.2'

shadow-support包提供了将Fragment主动添加到Activity中的方法:SupportFragmentTestUtil.startFragment(),简易的测试代码如下

Instrumentation

Android Instrumentation简介

Instrumentation框架主要是依靠控件的ID来进行定位的,拥有成熟的用例管理系统,是Android主推的白盒测试框架。若想对项目进行深入的、系统的单元测试,基本上都离不开Instrumentation这把屠龙宝刀。

参考文档

Android单元测试框架Robolectric3.0介绍(一)
使用强大的 Mockito 测试框架来测试你的代码
FaceBook 开源的一个测试网络访问速度的库,适用于安卓 APP
Android Espresso 测试框架介绍
在Android Studio中进行单元测试和UI测试
Android单元测试研究与实践
Android developer 测试支持库
要了解android单元测试,首先必须了解junit
Android单元测试框架Robolectric3.0介绍(一)
关于安卓单元测试,你需要知道的一切