通常我們?cè)趯懸粋€(gè)擴(kuò)展函數(shù),如果傳入一個(gè)lambda表達(dá)式大概是下面這樣
inline fun <T> T.apply(block: () -> Unit) {
block()
}
這個(gè)時(shí)候它的上下文是誰呢?
在擴(kuò)展的方法內(nèi)部,這個(gè)方法的上下文其實(shí)就是這個(gè)T,也就是調(diào)用這個(gè)方法的對(duì)象,所以this == T,這一點(diǎn)很明顯。
那么在外部使用的時(shí)候,如下代碼中呢?
val textView = TextView(activity).apply {
//這里的上下文是誰呢?
}
注意,這里雖然調(diào)用了apply方法,但是后面的代碼塊{}實(shí)際上是block: () -> Unit這個(gè)lambda的簡(jiǎn)寫,也就是說我們是處于一個(gè)lambda表達(dá)式中,所以它的上下文自然就是這段代碼外部的上下文??赡苁莂ctivity,也可能是別的什么東西。我在這里卡了一會(huì)兒,就是不小心把這里也當(dāng)做了擴(kuò)展的內(nèi)部,糾結(jié)于為毛這里不能拿到這個(gè)TextView對(duì)象。
但是我們知道Kotlin自己帶的apply方法是可以在這個(gè)lambda中去拿到調(diào)用它的對(duì)象,然后去做一些配置操作的,也就是說是可以把這個(gè)lambda中的上下文變成調(diào)用者的。怎么做呢?
其實(shí)很簡(jiǎn)單,將apply中的那個(gè)lambda也變成調(diào)用者的一個(gè)擴(kuò)展方法就行了。像下面這樣寫。
inline fun <T> T.apply(block: T.() -> Unit) {
block()
}
這里可以看到我給把這個(gè)lambda變成了T.()->Unit,表示它是T的一個(gè)擴(kuò)展方法。既然這個(gè)lambda一家是T的一個(gè)擴(kuò)展方法了,那么他內(nèi)部的上下文自然就是這個(gè)T了。所以就可以有了如下的寫法。
TextView(activity).apply {
text = "HelloDSl"
textSize = 16.sp2px()
}
是不是已經(jīng)和Kotlin自己的apply方法幾乎一樣了,但是既然apply是用來快速構(gòu)造對(duì)象的,那么自然需要它能將調(diào)用它的對(duì)象返回回來,所以加上返回值就變成了下面這樣。
inline fun <T> T.apply(block: T.() -> Unit): T {
block()
return this
}
val textView = TextView(activity).apply {
text = "HelloDSl"
textSize = 16.sp2px()
}
那么這個(gè)東西有什么用處呢?有了這個(gè)就可以輕松實(shí)現(xiàn)DSL啦!