Kotlin 写 Android 单元测试(四),Robolectric 在 JVM 上测试安卓相关代码

Kotlin 写 Android 单元测试系列:

Kotlin 写 Android 单元测试(一),单元测试是什么以及为什么需要

Kotlin 写 Android 单元测试(二),JUnit 4 测试框架和 kotlin.test 库的使用

Kotlin 写 Android 单元测试(三),Mockito mocking 框架的使用

Kotlin 写 Android 单元测试(四),Robolectric 在 JVM 上测试安卓相关代码

通过前面几篇文章,我们知道可以使用 JUnit 4 和 Mockito 测试框架来测试纯 Java 业务逻辑,但是无法在 JVM 上测试 Android 相关代码。因为 Android 代码需要运行在 Android 平台的虚拟机 Dalvik 或 ART 上,不能直接在 Java 虚拟机(JVM)上直接运行。而我们用 Android Studio 编写 Android 代码时只需要下载 JDK 和 Android SDK,在项目的External Libraries可以看到编译所需要的 Android API,实际上就是 android.jar,这样 Android 代码就能正常开发编译了。

但是 android.jar 里的类只是个壳,里面的方法都是throw RuntimeException("stub!!");,所以在 Android Studio 中可以正常开发编译,但是在 JDK 中 JVM 下运行的话会抛 RuntimeException。我们的 app 代码能在 Dalvik 或 ART 上运行,可以运行是因为它们把 android.jar 里面替换为 Android 的系统实现,所以才能正常运行。

这种情况如何做 Android 代码的单元测试呢?一种方式是使用 Android 官方提供的 Instrumentation 框架,不过测试代码还是不能在 JVM 上运行,只能在模拟器或者真机上运行,这种方式相当于编一个测试版的 apk,传到模拟器或者真机上再运行,显然速度不可能快,不方便做单元测试。那有没有什么方式可以直接 Android Studio 的开发环境 JVM 中运行 Android 代码呢,Robolectric 框架就可以解决这个问题。

本文是基于 Robolectric 3.5.1,Android Studio 3.0 环境

1. Robolectric

官网:http://robolectric.org/

Robolectric 重新实现了 android.jar,使得 Android 相关的测试代码都可以直接在 JVM 上运行。对于 Android 中的类XXX,它们都实现了ShadowXXX,例如 ShadowActivity、ShadowView 等,在调用 Activity 代码,Robolectric 会拦截并实际上调用 ShadowActivity 的同名方法。而且 Robolectric 还处理了布局加载、资源加载等 Android 运行需要的东西,可以在 JVM 中测试 Android 代码在真机上运行的样子。

1.1 Gradle 引入

testImplementation 'junit:junit:4.12'
testImplementation 'org.robolectric:robolectric:3.5.1'
// 如果用到 multidex 和 support-v4 包的话,还需要引入 robolectric 对应的模块
testImplementation 'org.robolectric:shadows-multidex:3.5.1'
testImplementation 'org.robolectric:shadows-supportv4:3.5.1'

android {
  testOptions {
    unitTests {
      includeAndroidResources = true
    }
  }
}  

注意:Robolectric 3.3 以上的版本都需要 Android Studio 的版本在 3.0 以上。

如果是 Linux 或者 Mac 用户,还需要在 Run -> Edit Configuration... 的窗口中,在左侧边栏选择 Defaults -> Android JUnit,然后将右侧的 working directory 值改为$MODULE_DIR$

1.2 测试实例

下面这个 Activity 点击按钮后会跳转到登录页。

class WelcomeActivity : Activity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.welcome_activity)
        
        findViewById(R.id.login).setOnClickListener { startActivity(Intent(this, LoginActivity::class.java)) }
    }
}

再看下测试代码:

@RunWith(RobolectricTestRunner::class)
class WelcomeActivityTest {

    @Test
    fun clickLogin() {
        val activity = Robolectric.setupActivity(WelcomeActivity::class.java)
        activity.findViewById(R.id.login).performClick()
        
        val expectedIntent = Intent(activity, LoginActivity::class.java)
        val actualIntent = ShadowApplication.getInstance().nextStartedActivity
        assertEquals(expectedIntent.component, actualIntent.component)
    }
}

上面测试代码中@RunWith(RobolectricTestRunner::class)保证 Robolectric 框架生效,这样才能在调用 Android 代码时转到 Robolectric 的 Shadow 实现。而且 Robolectric 会在测试框架执行一开始创建 application 实例,在 AndroidManifest.xml 中定义的 Application 类也会被反射创建实例,并执行 onCreate 生命周期。

ShadowApplication.getInstance().nextStartedActivity 是 Robolectric 提供方便测试的方法,可以获取最近一个启动的 Activity 的 Intent。

1.3 测试版的 Application

Robolectric 会自动识别出 AndroidManifest.xml 中定义的 Application 类,并且初始化,但是我们在做单元测试的时候,只需要测试项目代码的逻辑,不测试第三方库,例如网络请求、图片加载、数据库读写。我们测试的是调用第三方库时,所传递的参数是否符合预期,所以不需要初始化第三方库,而且 Robolectric 不支持加载第三方 native 库。

写单元测试时,有时需要使用 Mockito 框架 Mock 出一些对象,如果使用 Dagger 2 依赖注入框架的话,最好直接把 Module 里的依赖对象换成 Mock 对象,这时就需要改写 Application 的初始化逻辑。

值得庆幸的是,Robolectric 支持测试版的 Application,方法也很简单,所以可以用测试版的 Application 来修改初始化逻辑。如果在 AndroidManifest.xml 中定义的为 MyApplication,那么在测试文件下,同样包名,新建一个加上Test前缀的 Application 即可:

class TestMyApplication : MyApplication(), TestLifecycleApplication {
    override fun onCreate() {
        ...
    }

    override fun beforeTest(method: Method) {
    }

    override fun prepareTest(test: Any) {
    }

    override fun afterTest(method: Method) {
    }
}

这样 Robolectric 就好创建 TestMyApplication 实例作为应用的 Application。

2. 小结

总的来说,Robolectric 让我们很方便地测试 Android 相关代码,不过在一开始时使用时会遇到一些问题,坚持一下就过去了。使用过程中遇到什么问题,可以在网络下搜索下,或者断点看源码,当然可以在文章下面留言。我在使用过程就遇到一个问题:Application 实例创建后,Robolectric 会先注册广播监听器,然后在调用 Application 的 onCreate 方法,因为项目中的广播监听器初始化时用到了 onCreate 执行后才会生成的属性,结果就抛出空异常。

到目前为主,已经介绍了大体上在 Kotlin 下写 Android 单元测试需要用到的 JUnit、Mockito、Roblectric 测试框架,之后会写一些实际项目过程中单元测试的常见问题的解决方案。

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