kotlin作用域函数let run whith apply also 的区别takeIf 的用法

本质区别

有两个主要区别:

引用上下文对象的方式

thisit 使用this则具备上下文,可以省略thisit不能省略了,因为它是参数.

返回值

apply also返回上下文对象。
let,run, with返回 lambda 结果。

letrun

作用域函数不会引入任何新的技术功能,但它们可以使您的代码更加简洁和可读。
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

你可能感兴趣的:(kotlin作用域函数let run whith apply also 的区别takeIf 的用法)