我想说的是,不要放弃,只要你想单测的,都可以做到。
有疑惑看这里。
本文就是一篇解答疑惑的文章。
先看下基础知识
- 单测框架很多(Junit,Mockito,PowerMock,Mockk,Robolectric),但是他们大多数是重复的,只要会一个,其他的就都会了(差不多)。
- 单测的主要内容是什么,我自己总结了一下,就是 mock,verify,assert
- 单测中经常会有一些问题,静态方法怎么测?私有变量怎么修改?
Junit:
都是基础功能,看下别的文章一下就会了。
Mockito
都是基础的功能,它可以做什么呢?
-
mock(), spy()
mock 对象 -
verify()
及其变体:验证方法调用次数、执行顺序 -
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<String>(skuOrderApi,"2")
Whitebox.invokeMethod<Void>(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<Car>(relaxed = true)
object
的测试:mockkObject
mock 构造方法,mock 新创建的对象
mockkConstructor(MockCls::class)
every { anyConstructed<MockCls>().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 和逻辑一定要分开。
- 想要测试抽象类,可以写一个测试用的子类,然后对子类进行测试。