函数参数和对象的Kotlinunit testing
在Kotlin中,我们可以将函数对象作为函数参数传入。
- 如何创建unit testing来测试函数对象逻辑? (例如下面的
funcParam
) -
如何对具有函数参数的函数进行unit testing? (例如下面的
functionWithFuncParam
) – 即我可以创建一个模拟funcParam
?class MyClass1(val myObject: MyObject) { val funcParam = fun (num: Int): Int { return num * num } fun myFunctionOne() { myObject.functionWithFuncParam(funcParam) } } class MyObject () { fun functionWithFuncParam(funcParam: (Int) -> Int) { println(funcParam(32)) } }
假设funcParam
是public
你可以像其他方法一样测试它:
class MyClass1Tests { val sut = MyClass1(MyObject()) @Test fun `funcParam multiplies input`() { assertThat(sut.funcParam(4), equalTo(16)) assertThat(sut.funcParam(1), equalTo(1)) assertThat(sut.funcParam(0), equalTo(0)) assertThat(sut.funcParam(-10), equalTo(100)) } }
如果funcParam
是私有的,你不应该直接测试它的行为,而只能通过它的包含类的公共接口来测试它的行为。
当测试functionWithFuncParam
你可以很容易地提供(Int) -> Int
的存根实现:
class MyObjectTests { val outContent = ByteArrayOutputStream().apply { System.setOut(PrintStream(this)) } val sut = MyObject() @Test fun `functionWithFuncParam prints function output `() { sut.functionWithFuncParam { 12345678 } assertThat(outContent.toString(), containsString("12345678")) } }
如果你想测试MyClass1
与MyObject
交互,一种方法是使用MyClass1
接口实现的MyObject
。 如果两个class级是不同的合作者,通常是最好的选择,因为他们有一个独立的大多数不相关的行为:
interface FunctionalObj { fun functionWithFuncParam(funcParam: (Int) -> Int) } class MyClass1(val myObject: FunctionalObj) { //omitted for brevity } class MyClass1Tests { var params = mutableListOf<(Int)->Int>() val sut = MyClass1(object: FunctionalObj { override fun functionWithFuncParam(funcParam: (Int) -> Int) { params.add(funcParam) } }) @Test fun `myFunctionOne calls delegate`() { sut.myFunctionOne() assertThat(params.size, equalTo(1)) assertThat(params[0], equalTo(sut.funcParam))//only if `funcParam` is public } }
如果MyClass1
和MyObject
交互更复杂(即涉及更多的查询和命令调用),这意味着他们是同行密切合作。 在这种情况下,使用模拟会导致测试变得脆弱和困难。