91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Kotlin中標準函數run、with、let、also與apply有哪些區別

發布時間:2021-06-17 14:36:50 來源:億速云 閱讀:370 作者:小新 欄目:移動開發

這篇文章給大家分享的是有關Kotlin中標準函數run、with、let、also與apply有哪些區別的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

前言

和Java相比,在Kotlin中提供了不少新的特性。這次我們就來聊一聊Kotlin的一些通用的擴展標準函數run,with,let,also和apply。對于這五個標準函數它們都存在于Kotlin的源碼標準庫當中,也就是在Standard.kt文件當中。它們都是適用于任何對象的通用擴展函數。但是對于run,with,let,also和apply這五個函數他們的用法及其相似,以至于我們無法確定去選擇使用哪一個。那么現在我們就來聊一下這五個函數它們的使用方法,它們的不同之處以及在什么場景下去使用。

作用域函數

在這里我們重點是看一下run,with,T.run,T.let,T.also,和T.apply,對于這幾個函數來說它們最重要的功能之一是在調用函數的內部又提供了一個作用域。

那么下面就通過一段代碼來看一下run函數的作用域,對于其它函數來說當然也是類似。

fun test(){
 var animal = "cat"
 run {
  val animal = "dog"
  println(animal) // dog
 }
 println(animal)  //cat
}

在這個簡單的test函數當中我們擁有一個單獨的作用域,在run函數中能夠重新定義一個animal變量,并且它的作用域只存在于run函數當中。

目前對于這個run函數看起來貌似沒有什么用處,但是在run函數當中它不僅僅只是一個作用域,他還有一個返回值。他會返回在這個作用域當中的最后一個對象。

例如現在有這么一個場景,用戶領取app的獎勵,如果用戶沒有登錄彈出登錄dialog,如果已經登錄則彈出領取獎勵的dialog。我們可以使用以下代碼來處理這個邏輯。

run {
 if (islogin) loginDialog else getAwardDialog
}.show()

可以看到上面這段代碼會變得更加的簡潔,并且可以將show方法一次應用到上面兩個dialog當中,而不是去調用兩次。

with和其它通用標準函數

在這里之所以將with函數單獨拿出來進行說明,是因為with得用法和其它通用的標準函數的用法比較獨特。在這里我們依然使用run函數來進行對比。對于下面這段代碼做的是同樣一件事。它們的不同之處就是一個使用了with(T)函數,而另一個則是使用了T.run函數。

with(webView.settings){
 javaScriptEnabled = true
 databaseEnabled = true
}
webView.settings.run { 
 javaScriptEnabled = true
 databaseEnabled = true
}

但是我們覺得使用哪一個會更好呢?現在假設一種場景,那就是webView.settings可能為null。那我們就來再次看一下下面這段代碼.

with(webView.settings){
 javaScriptEnabled = true
 databaseEnabled = true
}
webView.settings?.run { 
 javaScriptEnabled = true
 databaseEnabled = true
}

這么以來就很明顯了,當然是T.run方法會更好,因為我們可以在使用這些函數之前可以進行對null的檢查。

對于with也是存在一個返回值,它也是會返回在這個作用域當中的最后一個對象。

作用域中接收者this和it

在這幾個擴展函數當中,它們都能直接獲取到調用的對象或者是with中傳入參數的對象。在這五個擴展函數在它們的作用域中的接收者可以是this或者是it。那么我們來對比一下T.run和T.let函數。這兩個函數也是十分的相似。

stringVariable?.run {
 println("字符串的長度為$length")
}

stringVariable?.let {
 println("字符串的長度為 ${it.length}")
}

在這兩段代碼中可以清晰的看到。在T.run函數中通過this來獲取stringVariable對象,而在T.let函數中通過it來取出stringVariable對象。當然我們也能夠為it重新命名。如果我們不想覆蓋外部作用域的this,這時候去使用T.let會更加的方便。至于哪些函數的接收者是this,哪些函數的接收者是it,在后面會通過一張樹狀圖清晰的體現出來。

在作用域中返回值的類型

在這些作用域中它們都會存在一個返回值。在上面的講述的run,with,T.run,T.let中它們返回的都是作用域中最后一個對象。當然它們所返回的值是允許和接受者it或者this對象的類型不同。但是并不是所有的標準函數都是返回作用域的最后一個對象。例如T.also函數。

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不管調用多少次返回的都是原來的original對象。

對于T.let和T.also都能夠進行鏈式操作,那么我們現在結合一下T.let和T.also的鏈式調用來看一下在實際場景中的應用。

//原始函數
fun makeDir(path: String): File {
 val result = File(path)
 result.mkdirs()
 return result
}

//通過let和also的鏈式調用改進后的函數
fun makeDir(path: String) = path.let{ File(it) }.also{ it.mkdirs() }

擴展函數的特性

到目前為止除了T.apply沒有使用到以外,根據上面的用法我們可以總結出來這些標準函數的三大特性。

  • 它們都有自己的作用域

  • 它們作用域中的接收者是this或者it

  • 它們都有一個返回值,返回最后一個對象(this)或者調用者自身(itself)

由此可想到對于T.apply無非也就是這三個特性。對于T.apply它作用域中的接收者是this,并且返回的調用者T。因此,T.apply的其中一個使用場景可以用來創建一個Fragment,代碼如下所示:

// 使用普通的方法創建一個Fragment
fun createInstance(args: Bundle) : MyFragment {
 val fragment = MyFragment()
 fragment.arguments = args
 return fragment
}

// 通過apply來改善原有的方法創建一個Fragment
fun createInstance(args: Bundle) 
    = MyFragment().apply { arguments = args }

我們也能夠通過T.apply的鏈式調用創建一個Intent:

// 普通創建Intent方法
fun createIntent(intentData: String, intentAction: String): Intent {
 val intent = Intent()
 intent.action = intentAction
 intent.data=Uri.parse(intentData)
 return intent
}

// 通過apply函數的鏈式調用創建Intent
fun createIntent(intentData: String, intentAction: String) =
  Intent().apply { action = intentAction }
    .apply { data = Uri.parse(intentData) }

如何選擇使用

在這里我們通過一個樹狀圖來看一下對著五個標準函數的區別,使用以及如何選取標準函數(圖片來源于參考文獻當中)

Kotlin中標準函數run、with、let、also與apply有哪些區別

總結

在這里做一下總結,我們可以看出在這五個通用標準函數當中它們的特性也是十分的簡單,無非也就是接收者和返回值的不同。對于with,T.run,T.apply接收者是this,而T.let和T.also接受者是it;對于with,T.run,T.let返回值是作用域的最后一個對象(this),而T.apply和T.also返回值是調用者本身(itself)。

感謝各位的閱讀!關于“Kotlin中標準函數run、with、let、also與apply有哪些區別”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

平塘县| 华阴市| 淳化县| 遂川县| 新干县| 龙岩市| 赤壁市| 正安县| 芜湖县| 平昌县| 平果县| 聊城市| 喀什市| 宁陕县| 湟源县| 宁化县| 枞阳县| 华池县| 崇阳县| 大关县| 铜鼓县| 双流县| 化隆| 华池县| 汝南县| 延安市| 通榆县| 盐城市| 蒙阴县| 夹江县| 仁布县| 通化县| 东丰县| 城固县| 文山县| 清徐县| 佛教| 荔浦县| 若尔盖县| 双峰县| 宜城市|