你可以使用 mockk()
来代替任何mock对象,比如说一个参数,下面我们要监听当执行调用某个方法时,返回一个 Car 实例:
every { … } return Car()
如果我们不想这样做(因为可能会因为实例化太麻烦),可以这样写:
every { … } return mockk()
我们可以在 mockk<T>()
构建时,填入一些参数,它的构造方法可填参数有这些:
inline fun mockk(
name: String? = null,
relaxed: Boolean = false,
vararg moreInterfaces: KClass<*>,
relaxUnitFun: Boolean = false,
block: T.() -> Unit = {}
) {…}
简单列下它们的作用:
-
name
: mock对象的名称 -
relaxed
: 是否对其代码进行依赖,默认为否,这个参数比较关键,后续会更细的讲解一下 -
moreInterfaces
: 让这个mock出来的对象实现这些声明的接口 -
relaxUnitFun
:和relaxed
差不多,但是只针对于 返回值是Unit 的方法, 后续会讲解一下 -
block
: 该语句块表示你在创建完 mock 对象后的操作,相当于.also{ ... }
语句
2.3.2 relaxed 和 relaxUnitFun
在 mock 一个对象时,这两个参数的意义是什么呢? 举个例子,我现在有一个被测类 Car,它依赖于一个 Engine:
class Car(private val engine: Engine) {
fun getSpeed(): Int {
return engine.getSpeed()
}
}
class Engine {
fun getSpeed(): Int {
return calSpeed()
}
private fun calSpeed(): Int {
return 30
}
}
我们要测试 getSpeed()
,它依赖于 Engine 里的方法,所以我们需要 mockk 一下 Engine,那么写下下面的测试方法:
fun testCar() {
// mock engine对象
val engine = mockk()
val car = Car(engine)
// 这里是私有方法设置监听的写法:
every { engine"calSpeed" } returns 30
val speed = car.getSpeed()
assertEquals(speed, 30)
}
但是这里我们报了一个错误: io.mockk.MockKException: no answer found for: Engine(#1).getSpeed()
这是因为mockk是严格去执行每个方法,而 Engine虽然mock了出来,但是mockk并不知道 Engine.getSpeed()
需不需要往下执行,所以它抛出了一个错误。
这个时候,你有两种解决方案。</