關(guān)于Kotlin擴(kuò)展函數(shù)與lambda的上下文

通常我們?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啦!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容