本質(zhì)區(qū)別
有兩個(gè)主要區(qū)別:
引用上下文對(duì)象的方式
this
和it
使用this
則具備上下文,可以省略this
而it
不能省略了,因?yàn)樗菂?shù).
返回值
apply
also
返回上下文對(duì)象稚机。
let
,run
, with
返回 lambda 結(jié)果弓坞。
let
與run
作用域函數(shù)不會(huì)引入任何新的技術(shù)功能蜡秽,但它們可以使您的代碼更加簡(jiǎn)潔和可讀。
如ListOf("1","2").let(it.add())
= var list=ListOf("1","2") list.add()
就是個(gè)語(yǔ)法糖,讓你寫(xiě)代碼好像更簡(jiǎn)潔,沒(méi)學(xué)過(guò)的朋友就以為是什么高大上的東西..
函數(shù) | 對(duì)象引用 | 返回值 | 是否是擴(kuò)展函數(shù) |
---|---|---|---|
let | it | Lambda 結(jié)果 | 是 |
run | this | Lambda 結(jié)果 | 是 |
run | this | Lambda 結(jié)果 | 否: 沒(méi)有上下文對(duì)象調(diào)用 |
with | this | Lambda 結(jié)果 | 否: 接受上下文對(duì)象作為參數(shù)甘耿。 |
apply | this | Context object | 是 |
also | it | Context object | 是 |
簡(jiǎn)短說(shuō)明
在非空對(duì)象上執(zhí)行 lambda:let
將表達(dá)式作為局部作用域中的變量引入:let
對(duì)象配置: apply
對(duì)象配置和計(jì)算結(jié)果: run
需要表達(dá)式的運(yùn)行語(yǔ)句:非擴(kuò)展 run
附加效果: also
兩個(gè)值替換
對(duì)對(duì)象進(jìn)行分組函數(shù)調(diào)用: with
作用域函數(shù)使用以下兩種方式之一
訪問(wèn)上下文對(duì)象:作為 lambdathis
或作為 lambda 參數(shù) it
。兩者都提供相同的功能竿滨,因此我們將針對(duì)不同情況描述各自的優(yōu)缺點(diǎn)
fun main() {
val str = "Hello"
// this
str.run {
// println("The receiver string length: $length")
println("The receiver string length: ${this.length}") //沒(méi)有區(qū)別
}
// it
str.let {
println("The receiver string's length is ${it}")
}
}
also
fun getRandomInt(): Int {
return Random.nextInt(100).also { value ->
// writeToLog("getRandomInt() generated value $value")
print("getRandomInt() generated value $it")
}
}
val i = getRandomInt()
println(i)
先打印日志內(nèi)容,getRandomInt() generated value 然后再打印i.
var numbers = mutableListOf("吉a", "兇bb", "以ccc", "情dddd", "遷eeeee")
println(numbers)
//return 由于執(zhí)行了map filter ,所以 返回了新的list對(duì)象 numbers本身不變.
var number2=numbers.map { it.length }.filter { it > 1 }.apply{println("apply $this")}
//return lamba result 這里是println輸出的是kotlin.Unit 如果再加上 ; it 那么和apply效果一樣了.
var number3=numbers.map { it.length }.filter { it > 1 }.let{println("let $it")}
println(number2)
println(number3)
輸出結(jié)果
[吉a, 兇bb, 以ccc, 情dddd, 遷eeeee]
apply [2, 3, 4, 5, 6]
let [2, 3, 4, 5, 6]
[2, 3, 4, 5, 6]
kotlin.Unit
如果只有一個(gè)it
參數(shù)可以直接使用.let(::println)
例子 with
val numbers = mutableListOf("one", "two", "three")
val firstAndLast = with(numbers) {
"The first element is ${this.first()}," +
" the last element is ${last()}"
}
println(firstAndLast)
執(zhí)行結(jié)果
The first element is one, the last element is three
takeIf
為真返回lambda結(jié)果,不匹配則返回null
takeUnless
是取反.
val number = Random.nextInt(100)
val evenOrNull = number.takeIf { it % 2 == 0 }
val oddOrNull = number.takeUnless { it % 2 == 0 }
println("even: $evenOrNull, odd: $oddOrNull")
結(jié)果,當(dāng)隨機(jī)值為33,那么takeif中是否等于偶數(shù)不成立,返回null,而takeUnless不成立就取lambda結(jié)果
even: null, odd: 33
騷操作 實(shí)現(xiàn)字符串如果不為空就轉(zhuǎn)換為大寫(xiě),否則整個(gè)結(jié)果返回null
val str = "Hello"
val caps = str.takeIf { it.isNotEmpty() }?.uppercase()
//val caps = str.takeIf { it.isNotEmpty() }.uppercase() //compilation error
println(caps)
參考連接
https://kotlinlang.org/docs/scope-functions.html#distinctions