Kotlin DSLs
Kotlin非常适合编写自己的类型安全构建器,也称为领域特定语言。正如你提到的,Anko库是使用DSL的一个例子。这里你需要理解的最重要的语言特性叫做“带接收者的函数字面值”,你已经使用过了:Test.() -> Unit
带接收者的函数字面值-基础知识
Kotlin支持“带接收者的函数字面值”概念。它使得在函数字面值体内可以无需任何特定限定符来调用接收者上的可见方法。这与扩展函数非常相似,在扩展函数中也可以访问扩展对象的成员。
一个简单的例子,也是Kotlin标准库中最酷的函数之一,就是apply
:
public inline fun <T> T.apply(block: T.() -> Unit): T { block(); return this }
正如您所看到的,此处将带有接收器的函数文字作为参数block
。这个block
只是被执行,并返回接收器(它是T
的一个实例)。实际操作如下:
val text: String = StringBuilder("Hello ").apply {
append("Kotliner")
append("! ")
append("How are you doing?")
}.toString()
一个StringBuilder
作为接收者并在其上调用apply
。{}
(lambda表达式)中作为参数传递的block
不需要使用其他限定符,只需多次调用StringBuilder
的可见方法append
。
具有接收者的函数字面值 - 在DSL中
如果您查看文档中的此示例,则可以看到它的实际效果:
class HTML {
fun body() { ... }
}
fun html(init: HTML.() -> Unit): HTML {
val html = HTML()
html.init()
return html
}
html {
body()
}
html()
函数期望一个带有 HTML
作为接收器的函数字面量。在函数体中,您可以看到它是如何使用的:创建了一个 HTML
实例并调用了其 init
方法。
好处
调用此类期望带有接收器的函数字面量(例如 html()
)的高阶函数的调用者,可以使用任何可见的 HTML
函数和属性而无需额外限定符(例如 this
),正如您在调用中所看到的那样。
html { // lambda with receiver begins here
body() // calling a method on the receiver object
}
你的示例
我创建了一个简单的示例,展示了您想要的内容:
class Context {
fun onSuccess(function: OnSuccessAction.() -> Unit) {
OnSuccessAction().function();
}
class OnSuccessAction {
fun toast(s: String) {
println("I'm successful <3: $s")
}
}
}
fun temp(function: Context.() -> Unit) {
Context().function()
}
fun main(args: Array<String>) {
temp {
onSuccess {
toast("Hello")
}
}
}