Android 单元测试从入门到放弃

我想说的是,不要放弃,只要你想单测的,都可以做到。
有疑惑看这里。
本文就是一篇解答疑惑的文章。

先看下基础知识

  1. 单测框架很多(Junit,Mockito,PowerMock,Mockk,Robolectric),但是他们大多数是重复的,只要会一个,其他的就都会了(差不多)。
  2. 单测的主要内容是什么,我自己总结了一下,就是 mock,verify,assert
  3. 单测中经常会有一些问题,静态方法怎么测?私有变量怎么修改?

Junit

都是基础功能,看下别的文章一下就会了。

Mockito

都是基础的功能,它可以做什么呢?

  1. mock(), spy() mock 对象
  2. verify() 及其变体:验证方法调用次数、执行顺序
  3. doReturn()、doThrow()、doAnswer()、doNothing()、doCallRealMethod() 拦截方法的返回值

特别的小知识

  • 方法连续调用测试
when(mockedUser.getName())
            .thenReturn("第一次调用")
            .thenThrow(new RuntimeException("第二次调用抛出异常了"))
            .thenReturn("第三次调用");

//或者
when(mockedUser.getName()).thenReturn("111", "222", "333");

  • 接口测试
    IPresenter mockPresenter = mock(IPresenter.class);
    when(mockPresenter.getUserName()).thenReturn("hanmeimei");
    String userName = mockPresenter.getUserName();

直接 mock 即可,和普通类一样

PowerMock

这个能力就强了,是 java 代码测试的主力。

特异功能:
对静态,final 方法,私有方法,构造方法都可以测试!

使用之前记的加上注解:
@RunWith(PowerMockRunner.class)
@PrepareForTest( { YourClassWithEgStaticMethod.class })

  • 私有方法/变量/构造函数的测试:

Whitebox.setInternalState(skuOrderApi,"dealId","1234")
Whitebox.getInternalState(skuOrderApi,"2")
Whitebox.invokeMethod(skuOrderApi, "onThirdPartyPaySuccess")
Whitebox.invokeConstructor(ZHSkuOrderApi::class.java)
  • 私有方法测试:

PowerMockito.when(underTest, "isExist").thenReturn(true);
  • 调用真正的方法,使用 thenCallRealMethod:

when(PaymentConfigInPlayChannel.canAccess()).thenCallRealMethod();
  • 静态方法的测试,使用 mockStatic
  • mock 系统的的静态方法类似,不一样的是要 @PrepareForTest()调用方的类,而不是系统的类
mockStatic(PaymentConfigInPlayChannel.class);
PowerMockito.whenNew(File.class).withArguments("bbb").thenReturn(file);

简单的原理解密
@PrepareForTest 会创建一个新的类,来代替原来的类。

Mockk

Mockk 是一个 kotlin 的单测框架,超级好用! https://github.com/mockk/mockk
对静态,final 方法,私有方法,构造方法都可以测试!

特异功能

  • relax mock,方法可以返回一些简单的值,使用方法:
    val car = mockk(relaxed = true)

  • object 的测试:mockkObject

  • mock 构造方法,mock 新创建的对象

mockkConstructor(MockCls::class)
every { anyConstructed().add(1, 2) } returns 4

  • mock 静态类、静态方法
mockkStatic(OrderParam::class)
every { OrderParam.ok() } returns "ok"

  • mock 接口
spyk(object :MyStateListener{})

  • mock 私有方法/变量 recordPrivateCalls
val myOrderApi: MyOrderApi = spyk(MyOrderApi(activity), recordPrivateCalls = true)
every { myOrderApi getProperty "price" } returns 33

verifyOrder {
    myOrderApi["toThirdPartyPay"]()
}

基础功能

  • verify(atLeast = 3) { car.accelerate(allAny()) } 次数验证
  • verifyOrder{ }顺序验证
  • verify(timeout = 3000){sum(1,2)} 设置超时等待时间的验证
  • ... ...

Robolectric

最常用的就是 mock 一个 Activity 了。虽然它最强大的是对 UI 进行测试,但是这里不建议,毕竟 UI 难测,需求变化快,不直观。我在开发过程中,也只用到了它的生命周期相关的功能,也是 UI 无关的。

activityController = Robolectric.buildActivity(TestActivity::class.java).create()
activity = activityController.get()

最后

一些零碎的建议:

  • 对于很简单的 java 代码,比如 model 类,用 Mockito 就可以了;复杂点的 java 代码,用 PowerMock;复杂点的 kotlin 代码,用 Mockk;需要用到 Activity 等对象的,可以引入 Robolectric 的生命周期功能。
  • SDK 在编写的初期,就要考虑单测,并且时间充裕的话可以边写单测边开发,防止之后为了单测重构代码。UI 和逻辑一定要分开。
  • 想要测试抽象类,可以写一个测试用的子类,然后对子类进行测试。

你可能感兴趣的:(Android 单元测试从入门到放弃)