当 Espresso 遇见 Android 单元测试

作者:施亮 团队:腾讯移动品质中心TMQ

引言

在Android单元测试中,不依赖Android环境的可以使用Junit。如果依赖Android环境,但是没有UI相关或者UI比较简单(如点击按钮)的单元测试可以使用开源库Robolectric解决依赖问题,使测试运行在JVM上,而非模拟器上,大大提高测试运行效率。

但是如果测试UI相关比较复杂的代码,又可以如何进行测试呢?

Activity& Espresso

Activity是承载UI控件的Android基本组件,Espresso则是可操作Activity的Google原生UI自动化框架。

Espresso特点有写法简单易入手,不可跨进程等(跨进程可使用Uiautomator),学习参考https://google.github.io/android-testing-support-library/docs/espresso/index.html。

工程中使用Espresso实现自动化测试只需要三步:

1、添加依赖:

androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2'

androidTestCompile'com.android.support.test:runner:0.5'

2、build.gradle中android.defaultConfig配置:

testInstrumentationRunner"android.support.test.runner.AndroidJUnitRunner"

3、写好case,进入工程主目录下执行gradlew connectedAndroidTest。

测试对象

如果项目是组件化开发的架构,将各个同类功能的代码整合在一个组件中,以便整体打包,便于维护,模块解耦合,持续构建单元测试等,可以减少底层修改导致上层错误的风险。

因此,UI控件库也作为一个单独的组件,比如时间选择器、标签组、数字选择器、带删除的输入框等等。

本文以控件时间选择器TimePicker作为测试对象来分析。

方案落地

想对复杂UI控件的代码进行单元测试,设计了如下方案。

1、将UI控件放入Activity(xml中配置);

2、添加一个输入框(也可用Spinner)和按钮用来提交命令,不同的命令控制UI控件调用不同的函数,Activity制作完成;

3、使用Espresso进行自动化操作输入命令和提交,并检查结果。

以上,UI控件执行了初始化代码及各函数代码,达到单元测试的目的。

那么问题来了,为何不用Espresso直接获取UI控件对象进行函数调用呢?因为Android更改UI只能在UI线程中进行,所以改变控件属性的代码只能写在Activity的代码中,而不是Espresso的测试代码中。

否则会报错: Only the original thread that created a view hierarchy cantouch its views.

Activity制作

由方案步骤1与2可知,需要制作一个Activity。

先看看带TimePicker和命令输入框的Activity实际效果:

时间选择器TimePicker的class文件的类结构:

分析类结构后,需要进行单元测试的函数为methodD()、methodE()、methodF()。其他函数为初始化或被测函数中调用的函数,都会被自动调用执行。

Activity中需要完成解析命令执行以上对应UI控件函数,根据输入框输入的字符来区分。

由此,Activity便制作完成。

Espresso闪亮登场

一切就绪,只欠Espresso。

Espresso需要做的事情,就是在已经做好的Activity提交不同的已定义命令,来执行UI控件不同的函数,并检查结果,达到单元测试目的。

1、以hideWeekDay(boolean hide)为例,函数功能为显示/隐藏星期的数字。

用例设计为控件显示星期与隐藏星期,即hideWeekDay(true)与hideWeekDay(false),如隐藏星期的显示,则步骤为:

(1)命令输入框输入hideWeek(已在Activity中做好解析);

(2)点击提交按钮;

(3)检查年正常显示;

(4)检查月正常显示;

(5)检查日正常显示;

(6)检查星期未显示,已隐藏。

测试代码如下:

若隐藏星期显示时,却隐藏了年显示,则会报错。

由此,hideWeekDay(boolean hide)单元测试完成。

2、再以getViewDate()为例,函数功能为获取当前控件显示的时间。

Activity中解析到输入为getViewDate命令时,执行show.setText(mTimePicker.getViewDate()),将获取的UI控件当前时间显示在了id为show的TextView上,以便Espresso检查。

用例设计为获取当前控件显示的时间与系统的真实时间对比,并且逐个对比年、月、日、星期是否正确。测试代码如下:

整个TimePicker的测试就依此方式测试完成,测试类的结构:

测试报告

Activity与测试类写好后,PC连接一台真机,使用命令行进入工程根目录下,运行gradlewconnectedAndroidTest命令即可测试与输出测试报告。

html报告会自动生成,位于工程下build/reports/androidTests/connected/index.html。

总结

综上,不同的角色处理不同的事务。

Activity中显示UI控件、解析命令与调用UI控件函数。

Espr 当然,Espresso不仅局限于此,可以一定程度上取代Robolectric与Uiautomator。

并且,Espresso与Uiautomator依赖可以同时添加在工程中混合使用,也是棒棒哒。

获取更多测试干货,请搜索微信公众号腾讯移动品质中心TMQ!

原创声明,本文系作者授权云+社区-专栏发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

编辑于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏菩提树下的杨过

Silverlight:分包下载及SEO优化方案

一、按模块分包 一般大型的Silverlight应用,都会按模块分解成多个silverlight project,编译后就有多个xap包,然后在需要用到的场景按...

1775
来自专栏IMWeb前端团队

Nodejs进阶:如何将图片转成datauri嵌入到网页中去

本文作者:IMWeb 陈映平 原文出处:IMWeb社区 未经同意,禁止转载 问题:将图片转成datauri 今天,在QQ群有个群友问了个问题:“nod...

1748
来自专栏数据小魔方

数据地图系列6|Stata数据地图(下)

今天要跟大家分享的是数据地图系列6——Stata数据地图(下)! 接着前一篇的节凑,这一篇会给大家介绍比较全面的Stata热力地图代码实现。 版本仍然是基于S...

3574
来自专栏FSociety

Python网易云音乐爬虫进阶篇

年前写过一篇爬网易云音乐评论的文章,爬不了多久又回被封,所以爬下来那么点根本做不了什么分析,后面就再改了下,加入了多线程,一次性爬一个歌手最热门50首歌曲的评论...

752
来自专栏python+iOS学习交流

7年iOS架构师教你如何快速提高并掌握 iOS开发核心技能

首先你要花点时间针对objective-c语言的学习;毕竟这个是iOS开发的基础(你也可以尝试用Swift,但此项目只是针对OC),编程套路其实都是差不多,多写...

1291
来自专栏H2Cloud

游戏服务器设计之NPC系统

游戏服务器设计之NPC系统 简介 NPC系统是游戏中非常重要的系统,设计的好坏很大程度上影响游戏的体验。NPC在游戏中有如下作用: 引导玩家体验游戏内容,一般游...

3294
来自专栏Material Design组件

Human Interface Guidelines —— Refresh Content Controls

948
来自专栏菩提树下的杨过

Silverlight Telerik控件学习:RadTransitionControl

如果展示类似这种比较cool的图片轮换效果,用RadTransitionControl控件就对了,它提供的过渡效果非常cool! 原理并不复杂,可参见以前写的 ...

1857
来自专栏许斌盛的专栏

基于 ffmpeg 的跨平台播放器实现

随着游戏娱乐等直播业务的增长,在移动端观看直播的需求也日益迫切。但是移动端原生的播放器对各种直播流的支持却不是很好。Android 原生的 MediaPlaye...

2.1K0
来自专栏腾讯Bugly的专栏

Android APK 瘦身 - JOOX Music项目实战

导语 JOOX Music是腾讯海外布局的一个音乐产品,2014年发布以来已经成为5个国家和地区排名第一的音乐App。东南亚是JOOX Music的主要发行地区...

3126

扫码关注云+社区