Android单元测试最佳实践

目的

充分的单元测试就是提高代码质量最有效的手段之一,而单元测试严重依赖代码的可测试性,本文主要通过一个简单的DEMO演示如何对Android原生应用进行单元测试,同时示例代码采用MVP模式以提高代码的可读性和可测试性

简介

在Android原生应用开发中,存在两种单元测试:本地JVM测试和Instrumentation测试。本文仅介绍本地JVM测试

  • 本地jvm的单元测试
    这种方式运行速度快,对运行环境没有特殊要求,可以很方便的做自动化测试,是单元测试首选的方法

  • Instrumentation测试
    Instrumentation测试需要运行在Android环境下,可以是模拟器或者手机等真实设备。这种方式运行速度慢,且严重依赖Android运行环境,更适合用来做集成测试

准备

我准备了一个简单的APP,模拟一个耗时的网络请求获得一段数据并显示在界面上,针对这个APP编写单元测试用例并进行本地单元测试。

App运行效果

依赖库

依赖库 作用
JUnit-4.12 基础得单元测试框架
Robolectric-3.8 Android SDK测试框架
PowerMock-1.6.6 模拟被测对象依赖的静态方法
Mockito-1.10.19 模拟被测对象依赖的对象

配置build.gradle

  • 增加编译选项,在测试中包含资源文件
    testOptions {
        unitTests {
            includeAndroidResources true
        }
    }
  • 添加测试依赖库
    testImplementation 'junit:junit:4.12'
    testImplementation 'org.robolectric:robolectric:3.8'
    testImplementation 'org.robolectric:shadows-supportv4:3.8'
    testImplementation 'org.powermock:powermock-module-junit4:1.6.6'
    testImplementation 'org.powermock:powermock-module-junit4-rule:1.6.6'
    testImplementation 'org.powermock:powermock-api-mockito:1.6.6'
    testImplementation 'org.powermock:powermock-classloading-xstream:1.6.6'
    testImplementation 'org.mockito:mockito-all:1.10.19'

测试Activity

测试Activity主要是测试它各个生命周期的状态变化、对外界输入的响应是否符合预期,Activity测试完全依赖Android SDK,需要用Robolectric。

Robolectric是一个开源的单元测试框架,能够完全模拟Android SDK并在JVM中运行。

UI依赖于Persenter,在Activity中通过静态工厂方法创建依赖的Presenter实例,需要使用PowerMock来模拟创建Presenter过程,完成Presenter模拟对象的注入

配置

  • 通过@RunWith指定使用RobolectricTestRunner
  • 通过@Config配置Robolectric的运行环境
  • 通过@PrepareForTest配置PowerMock需要模拟的静态类型
@RunWith(RobolectricTestRunner.class)
@Config(sdk = 21, constants = BuildConfig.class)
@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})
@PrepareForTest({PresenterFactory.class})
    @Before
    public void setUp() {
        appContext = RuntimeEnvironment.application.getApplicationContext();
        PowerMockito.mockStatic(PresenterFactory.class);
    }

onCreate用例

通过Robolectric的ActivityController来构建并管理activity的生命周期,运行至onCreate阶段,然后验证这个阶段text1是否正确初始化

    @Test
    public void onCreate_text1() {
        MainActivity activity = Robolectric.buildActivity(MainActivity.class).create().get();
        String expect = appContext.getString(R.string.hell_world);
        assertEquals(expect, ((TextView)activity.findViewById(R.id.lbl_text1)).getText());
    }

Click Button1用例

Activity完全显示以后,验证button1的click操作是否显示toast消息

    @Test
    public void btn1_click() {
        MainActivity activity = Robolectric.setupActivity(MainActivity.class);
        activity.findViewById(R.id.btn_1).performClick();
        String expect = appContext.getString(R.string.hell_world);
        assertEquals(expect, ShadowToast.getTextOfLatestToast());
    }

Click Button2用例

Activity完全显示以后,验证button2的click操作是否调用了presenter的fetch方法

    @Test
    public void btn2_click() {
        MainContract.Presenter presenter = Mockito.mock(MainContract.Presenter.class);
        PowerMockito.when(PresenterFactory.create(Mockito.any(MainContract.View.class), Mockito.any(AppExecutors.class)))
                .thenReturn(presenter);

        MainActivity activity = Robolectric.setupActivity(MainActivity.class);

        activity.findViewById(R.id.btn_2).performClick();

        Mockito.verify(presenter, Mockito.times(1))
                .fetch();
    }

测试Presenter

Presenter的测试一般可以不用依赖Android SDK了,Presenter依赖于底层的领域服务,也依赖上层View,demo中对领域服务的依赖没有通过构造函数的方式注入,而是通过静态工厂方法构建,还是需要用到PowerMock

配置

  • 通过@RunWith指定使用PowerMockRunner
  • 通过@PrepareForTest配置PowerMock需要模拟的静态类型
@RunWith(PowerMockRunner.class)
@PrepareForTest({ServiceFactory.class})
    @Before
    public void setUp() {
        PowerMockito.mockStatic(ServiceFactory.class);
    }

成功路径用例

验证View的方法是否成功调用且调用参数是否一致

    @Test
    public void fetch_success() {
        String expected = "hello world";
        SlowService service = Mockito.mock(SlowService.class);
        Mockito.when(service.fetch()).thenReturn(expected);
        PowerMockito.when(ServiceFactory.create())
                .thenReturn(service);

        MainContract.View view = Mockito.mock(MainContract.View.class);
        MainPresenter presenter = new MainPresenter(view, executors);

        presenter.fetch();

        Mockito.verify(service, Mockito.times(1)).fetch();
        Mockito.verify(view, Mockito.times(1)).onFetchStarted();
        Mockito.verify(view, Mockito.times(1)).onFetchCompleted();
        Mockito.verify(view, Mockito.times(0)).onFetchFailed(Mockito.anyObject());
        ArgumentCaptor<String> captor = ArgumentCaptor.forClass(String.class);
        Mockito.verify(view, Mockito.times(1)).onFetchSuccess(captor.capture());
        assertEquals(expected, captor.getValue());
    }

失败路径用例

    @Test
    public void fetch_failed() {
        RuntimeException exception = new RuntimeException("fetch failed");

        SlowService service = Mockito.mock(SlowService.class);
        Mockito.when(service.fetch()).thenThrow(exception);
        PowerMockito.when(ServiceFactory.create())
                .thenReturn(service);

        MainContract.View view = Mockito.mock(MainContract.View.class);
        MainPresenter presenter = new MainPresenter(view, executors);

        presenter.fetch();

        Mockito.verify(service, Mockito.times(1)).fetch();
        Mockito.verify(view, Mockito.times(1)).onFetchStarted();
        Mockito.verify(view, Mockito.times(1)).onFetchCompleted();
        ArgumentCaptor<Throwable> captor = ArgumentCaptor.forClass(Throwable.class);
        Mockito.verify(view, Mockito.times(1)).onFetchFailed(captor.capture());
        assertEquals(exception, captor.getValue());
        Mockito.verify(view, Mockito.times(0)).onFetchSuccess(Mockito.anyString());
    }

测试Service

Service不会对上层有依赖,可以直接使用JUnit测试

public class SlowServiceImplTest {

    @Test
    public void fetch_data() {
        SlowServiceImpl impl = new SlowServiceImpl();
        String data = impl.fetch();
        assertEquals("from slow service", data);
    }

}

自动化测试

自动化测试一般是在持续集成环境中使用命令来执行单元测试

gradlew :app:testDebugUnitTest

总结

写完这个demo,总觉得给Android APP做单元测试还是非常简单的,作为一个优秀的程序员,怎么能够不关注自己的代码质量呢,还是自己动手试试吧

源码下载

https://github.com/hziee514/android-testing

参考资料

Robolectric
Using PowerMock
Mockito
PowerMock

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容