通常我們在寫一個擴(kuò)展函數(shù)完域,如果傳入一個lambda表達(dá)式大概是下面這樣
inline fun <T> T.apply(block: () -> Unit) {
block()
}
這個時候它的上下文是誰呢?
在擴(kuò)展的方法內(nèi)部计寇,這個方法的上下文其實(shí)就是這個T腾它,也就是調(diào)用這個方法的對象初嘹,所以this == T,這一點(diǎn)很明顯绵跷。
那么在外部使用的時候螺句,如下代碼中呢勾笆?
val textView = TextView(activity).apply {
//這里的上下文是誰呢阿宅?
}
注意候衍,這里雖然調(diào)用了apply方法,但是后面的代碼塊{}實(shí)際上是block: () -> Unit這個lambda的簡寫洒放,也就是說我們是處于一個lambda表達(dá)式中脱柱,所以它的上下文自然就是這段代碼外部的上下文±觯可能是activity,也可能是別的什么東西惨好。我在這里卡了一會兒煌茴,就是不小心把這里也當(dāng)做了擴(kuò)展的內(nèi)部,糾結(jié)于為毛這里不能拿到這個TextView對象日川。
但是我們知道Kotlin自己帶的apply方法是可以在這個lambda中去拿到調(diào)用它的對象蔓腐,然后去做一些配置操作的,也就是說是可以把這個lambda中的上下文變成調(diào)用者的龄句。怎么做呢回论?
其實(shí)很簡單,將apply中的那個lambda也變成調(diào)用者的一個擴(kuò)展方法就行了分歇。像下面這樣寫傀蓉。
inline fun <T> T.apply(block: T.() -> Unit) {
block()
}
這里可以看到我給把這個lambda變成了T.()->Unit,表示它是T的一個擴(kuò)展方法职抡。既然這個lambda一家是T的一個擴(kuò)展方法了葬燎,那么他內(nèi)部的上下文自然就是這個T了。所以就可以有了如下的寫法缚甩。
TextView(activity).apply {
text = "HelloDSl"
textSize = 16.sp2px()
}
是不是已經(jīng)和Kotlin自己的apply方法幾乎一樣了谱净,但是既然apply是用來快速構(gòu)造對象的,那么自然需要它能將調(diào)用它的對象返回回來擅威,所以加上返回值就變成了下面這樣壕探。
inline fun <T> T.apply(block: T.() -> Unit): T {
block()
return this
}
val textView = TextView(activity).apply {
text = "HelloDSl"
textSize = 16.sp2px()
}
那么這個東西有什么用處呢?有了這個就可以輕松實(shí)現(xiàn)DSL啦郊丛!