热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

详解Android单元测试最佳实践

这篇文章主要介绍了详解Android单元测试最佳实践,本文介绍了如何对Android原生应用进行单元测试,同时示例代码采用MVP模式以提高代码的可读性和可测试性

目的

充分的单元测试就是提高代码质量最有效的手段之一,而单元测试严重依赖代码的可测试性,本文主要通过一个简单的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

配置

  1. 通过@RunWith指定使用PowerMockRunner
  2. 通过@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 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 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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
  • GPT-3发布,动动手指就能自动生成代码的神器来了!
    近日,OpenAI发布了最新的NLP模型GPT-3,该模型在GitHub趋势榜上名列前茅。GPT-3使用的数据集容量达到45TB,参数个数高达1750亿,训练好的模型需要700G的硬盘空间来存储。一位开发者根据GPT-3模型上线了一个名为debuid的网站,用户只需用英语描述需求,前端代码就能自动生成。这个神奇的功能让许多程序员感到惊讶。去年,OpenAI在与世界冠军OG战队的表演赛中展示了他们的强化学习模型,在限定条件下以2:0完胜人类冠军。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 云原生边缘计算之KubeEdge简介及功能特点
    本文介绍了云原生边缘计算中的KubeEdge系统,该系统是一个开源系统,用于将容器化应用程序编排功能扩展到Edge的主机。它基于Kubernetes构建,并为网络应用程序提供基础架构支持。同时,KubeEdge具有离线模式、基于Kubernetes的节点、群集、应用程序和设备管理、资源优化等特点。此外,KubeEdge还支持跨平台工作,在私有、公共和混合云中都可以运行。同时,KubeEdge还提供数据管理和数据分析管道引擎的支持。最后,本文还介绍了KubeEdge系统生成证书的方法。 ... [详细]
  • 学习笔记(34):第三阶段4.2.6:SpringCloud Config配置中心的应用与原理第三阶段4.2.6SpringCloud Config配置中心的应用与原理
    立即学习:https:edu.csdn.netcourseplay29983432482?utm_sourceblogtoedu配置中心得核心逻辑springcloudconfi ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • svnWebUI:一款现代化的svn服务端管理软件
    svnWebUI是一款图形化管理服务端Subversion的配置工具,适用于非程序员使用。它解决了svn用户和权限配置繁琐且不便的问题,提供了现代化的web界面,让svn服务端管理变得轻松。演示地址:http://svn.nginxwebui.cn:6060。 ... [详细]
author-avatar
蓝色水气球_453
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有