前言
和Java相比,在Kotlin中提供了不少新的特性。這次我們就來聊一聊Kotlin的一些通用的擴(kuò)展標(biāo)準(zhǔn)函數(shù)run,with,let,also和apply。對于這五個標(biāo)準(zhǔn)函數(shù)它們都存在于Kotlin的源碼標(biāo)準(zhǔn)庫當(dāng)中,也就是在Standard.kt文件當(dāng)中。它們都是適用于任何對象的通用擴(kuò)展函數(shù)。但是對于run,with,let,also和apply這五個函數(shù)他們的用法及其相似,以至于我們無法確定去選擇使用哪一個。那么現(xiàn)在我們就來聊一下這五個函數(shù)它們的使用方法,它們的不同之處以及在什么場景下去使用。
作用域函數(shù)
在這里我們重點(diǎn)是看一下run,with,T.run,T.let,T.also,和T.apply,對于這幾個函數(shù)來說它們最重要的功能之一是在調(diào)用函數(shù)的內(nèi)部又提供了一個作用域。
那么下面就通過一段代碼來看一下run函數(shù)的作用域,對于其它函數(shù)來說當(dāng)然也是類似。
fun test(){ var animal = "cat" run { val animal = "dog" println(animal) // dog } println(animal) //cat}在這個簡單的test函數(shù)當(dāng)中我們擁有一個單獨(dú)的作用域,在run函數(shù)中能夠重新定義一個animal變量,并且它的作用域只存在于run函數(shù)當(dāng)中。
目前對于這個run函數(shù)看起來貌似沒有什么用處,但是在run函數(shù)當(dāng)中它不僅僅只是一個作用域,他還有一個返回值。他會返回在這個作用域當(dāng)中的最后一個對象。
例如現(xiàn)在有這么一個場景,用戶領(lǐng)取app的獎勵,如果用戶沒有登錄彈出登錄dialog,如果已經(jīng)登錄則彈出領(lǐng)取獎勵的dialog。我們可以使用以下代碼來處理這個邏輯。
run { if (islogin) loginDialog else getAwardDialog}.show()可以看到上面這段代碼會變得更加的簡潔,并且可以將show方法一次應(yīng)用到上面兩個dialog當(dāng)中,而不是去調(diào)用兩次。
with和其它通用標(biāo)準(zhǔn)函數(shù)
在這里之所以將with函數(shù)單獨(dú)拿出來進(jìn)行說明,是因?yàn)閣ith得用法和其它通用的標(biāo)準(zhǔn)函數(shù)的用法比較獨(dú)特。在這里我們依然使用run函數(shù)來進(jìn)行對比。對于下面這段代碼做的是同樣一件事。它們的不同之處就是一個使用了with(T)函數(shù),而另一個則是使用了T.run函數(shù)。
with(webView.settings){ javaScriptEnabled = true databaseEnabled = true}webView.settings.run { javaScriptEnabled = true databaseEnabled = true}但是我們覺得使用哪一個會更好呢?現(xiàn)在假設(shè)一種場景,那就是webView.settings可能為null。那我們就來再次看一下下面這段代碼.
with(webView.settings){ javaScriptEnabled = true databaseEnabled = true}webView.settings?.run { javaScriptEnabled = true databaseEnabled = true}這么以來就很明顯了,當(dāng)然是T.run方法會更好,因?yàn)槲覀兛梢栽谑褂眠@些函數(shù)之前可以進(jìn)行對null的檢查。
對于with也是存在一個返回值,它也是會返回在這個作用域當(dāng)中的最后一個對象。
作用域中接收者this和it
在這幾個擴(kuò)展函數(shù)當(dāng)中,它們都能直接獲取到調(diào)用的對象或者是with中傳入?yún)?shù)的對象。在這五個擴(kuò)展函數(shù)在它們的作用域中的接收者可以是this或者是it。那么我們來對比一下T.run和T.let函數(shù)。這兩個函數(shù)也是十分的相似。
stringVariable?.run { println("字符串的長度為$length")}stringVariable?.let { println("字符串的長度為 ${it.length}")}在這兩段代碼中可以清晰的看到。在T.run函數(shù)中通過this來獲取stringVariable對象,而在T.let函數(shù)中通過it來取出stringVariable對象。當(dāng)然我們也能夠?yàn)閕t重新命名。如果我們不想覆蓋外部作用域的this,這時候去使用T.let會更加的方便。至于哪些函數(shù)的接收者是this,哪些函數(shù)的接收者是it,在后面會通過一張樹狀圖清晰的體現(xiàn)出來。
在作用域中返回值的類型
在這些作用域中它們都會存在一個返回值。在上面的講述的run,with,T.run,T.let中它們返回的都是作用域中最后一個對象。當(dāng)然它們所返回的值是允許和接受者it或者this對象的類型不同。但是并不是所有的標(biāo)準(zhǔn)函數(shù)都是返回作用域的最后一個對象。例如T.also函數(shù)。
val original = "abc"original.let { println("The original String is $it") // "abc" it.reversed() }.let { println("The reverse String is $it") // "cba" it.length }.let { println("The length of the String is $it") // 3}original.also { println("The original String is $it") // "abc" it.reversed() }.also { println("The reverse String is ${it}") // "abc" it.length }.also { println("The length of the String is ${it}") // "abc"}從上面兩段代碼可以看出T.let和T.also的返回值使不同的。T.let返回的是作用域中的最后一個對象,它的值和類型都可以改變。但是T.also不管調(diào)用多少次返回的都是原來的original對象。
對于T.let和T.also都能夠進(jìn)行鏈?zhǔn)讲僮鳎敲次覀儸F(xiàn)在結(jié)合一下T.let和T.also的鏈?zhǔn)秸{(diào)用來看一下在實(shí)際場景中的應(yīng)用。
//原始函數(shù)fun makeDir(path: String): File { val result = File(path) result.mkdirs() return result}//通過let和also的鏈?zhǔn)秸{(diào)用改進(jìn)后的函數(shù)fun makeDir(path: String) = path.let{ File(it) }.also{ it.mkdirs() }擴(kuò)展函數(shù)的特性
到目前為止除了T.apply沒有使用到以外,根據(jù)上面的用法我們可以總結(jié)出來這些標(biāo)準(zhǔn)函數(shù)的三大特性。
由此可想到對于T.apply無非也就是這三個特性。對于T.apply它作用域中的接收者是this,并且返回的調(diào)用者T。因此,T.apply的其中一個使用場景可以用來創(chuàng)建一個Fragment,代碼如下所示:
// 使用普通的方法創(chuàng)建一個Fragmentfun createInstance(args: Bundle) : MyFragment { val fragment = MyFragment() fragment.arguments = args return fragment}// 通過apply來改善原有的方法創(chuàng)建一個Fragmentfun createInstance(args: Bundle) = MyFragment().apply { arguments = args }我們也能夠通過T.apply的鏈?zhǔn)秸{(diào)用創(chuàng)建一個Intent:
// 普通創(chuàng)建Intent方法fun createIntent(intentData: String, intentAction: String): Intent { val intent = Intent() intent.action = intentAction intent.data=Uri.parse(intentData) return intent}// 通過apply函數(shù)的鏈?zhǔn)秸{(diào)用創(chuàng)建Intentfun createIntent(intentData: String, intentAction: String) = Intent().apply { action = intentAction } .apply { data = Uri.parse(intentData) }如何選擇使用
在這里我們通過一個樹狀圖來看一下對著五個標(biāo)準(zhǔn)函數(shù)的區(qū)別,使用以及如何選取標(biāo)準(zhǔn)函數(shù)(圖片來源于參考文獻(xiàn)當(dāng)中)

總結(jié)
在這里做一下總結(jié),我們可以看出在這五個通用標(biāo)準(zhǔn)函數(shù)當(dāng)中它們的特性也是十分的簡單,無非也就是接收者和返回值的不同。對于with,T.run,T.apply接收者是this,而T.let和T.also接受者是it;對于with,T.run,T.let返回值是作用域的最后一個對象(this),而T.apply和T.also返回值是調(diào)用者本身(itself)。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網(wǎng)的支持。
新聞熱點(diǎn)
疑難解答
圖片精選