本质区别
有两个主要区别:
引用上下文对象的方式
this
和it
使用this
则具备上下文,可以省略this
而it
不能省略了,因为它是参数.
返回值
apply
also
返回上下文对象。
let
,run
, with
返回 lambda 结果。
let
与run
作用域函数不会引入任何新的技术功能,但它们可以使您的代码更加简洁和可读。
如ListOf("1","2").let(it.add())
= var list=ListOf("1","2") list.add()
就是个语法糖,让你写代码好像更简洁,没学过的朋友就以为是什么高大上的东西..
函数 | 对象引用 | 返回值 | 是否是扩展函数 |
---|---|---|---|
let | it | Lambda 结果 | 是 |
run | this | Lambda 结果 | 是 |
run | this | Lambda 结果 | 否: 没有上下文对象调用 |
with | this | Lambda 结果 | 否: 接受上下文对象作为参数。 |
apply | this | Context object | 是 |
also | it | Context object | 是 |
简短说明
在非空对象上执行 lambda:let
将表达式作为局部作用域中的变量引入:let
对象配置: apply
对象配置和计算结果: run
需要表达式的运行语句:非扩展 run
附加效果: also
两个值替换
对对象进行分组函数调用: with
作用域函数使用以下两种方式之一
访问上下文对象:作为 lambdathis
或作为 lambda 参数 it
。两者都提供相同的功能,因此我们将针对不同情况描述各自的优缺点
fun main() {
val str = "Hello"
// this
str.run {
// println("The receiver string length: $length")
println("The receiver string length: ${this.length}") //没有区别
}
// 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)
先打印日志内容,getRandomInt() generated value 然后再打印i.
var numbers = mutableListOf("吉a", "凶bb", "以ccc", "情dddd", "迁eeeee")
println(numbers)
//return 由于执行了map filter ,所以 返回了新的list对象 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)
输出结果
[吉a, 凶bb, 以ccc, 情dddd, 迁eeeee]
apply [2, 3, 4, 5, 6]
let [2, 3, 4, 5, 6]
[2, 3, 4, 5, 6]
kotlin.Unit
如果只有一个it
参数可以直接使用.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)
执行结果
The first element is one, the last element is three
takeIf
为真返回lambda结果,不匹配则返回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")
结果,当随机值为33,那么takeif中是否等于偶数不成立,返回null,而takeUnless不成立就取lambda结果
even: null, odd: 33
骚操作 实现字符串如果不为空就转换为大写,否则整个结果返回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