@Thomas Martin 的回答是正确的,但你问为什么它会有所作为。
In KotestSpringSpec
依赖于一些 Kotlin DSL 功能来工作。让我们通过从头开始构建相同的函数来进行演示。
我们将从使用 Kotlin 的扩展函数开始。这允许我们向我们无法控制的类添加函数。
所以,
fun String.test(test: () -> Unit)
那么我们可以在任何字符串上调用这个测试函数:
"this is a test".test({ 1 + 2 shouldBe 3 })
其次,Kotlin 允许将任何最终的 lambda arg 带到括号之外。所以foo(arg, arg2, arg3, lambda)
可以成为foo(arg, arg2, arg3) lambda
。这意味着我们可以将测试编写为:
"this is a test".test { 1 + 2 shouldBe 3 }
接下来,我们可以将函数标记为中缀,然后我们就不需要使用点来调用它们了。所以我们的测试函数变成了:
infix fun String.test(test: () -> Unit)
我们的测试现在看起来像:
"this is a test" test { 1 + 2 shouldBe 3 }
最后,任何命名invoke
并标记为操作符函数的函数都可以在没有函数名的情况下调用。因此operator fun invoke(a: String, b: String)
,在类 Foo 中可以作为常规Foo.invoke(a,b)
或仅调用Foo(a, b)
。
所以把所有这些放在一起,我们最终的测试函数看起来像:
operator fun String.invoke(test: () -> Unit)
我们的测试结果如下:
"this is a test" { 1 + 2 shouldBe 3 }
或者更有可能,
"this is a test" {
1 + 2 shouldBe 3
}
如果大括号移动到下一行,就像你原来的问题一样,它看起来就像一个字符串,后面跟着一个不相关的 lambda 块。两种不同的表达方式。