Kotlin-作用域函数

本文介绍了Kotlin中的一些作用域函数,如let,run,with,apply和also,它们用于在对象上下文中执行代码块。let常用于处理非空值,with和run在不需返回结果时操作对象,apply用于对象配置,also则用于在操作对象的同时保持对它的引用。这些函数能提高代码的简洁性和可读性,但也需要注意避免过度使用和混淆上下文。
摘要由CSDN通过智能技术生成

在对象的上下文中执行代码块。当您在提供lambda表达式的对象上调用这样的函数时,它会形成一个临时作用域。在此范围内,可以不带名称地访问对象。这样的函数称为作用域函数。
let run with apply also
作用域函数不会引入任何新的技术功能,但它们可以使代码更加简洁易读。
尽管作用域函数可以使代码更加简洁,但要避免过度使用它们:这会使代码难以阅读并导致错误。我们还建议避免嵌套作用域函数,并在链接它们时要小心,因为很容易混淆当前上下文对象和this或it的值。

对比
函数对象引用返回值是否是扩展函数
letitLambda resultYes
runthisLambda resultYes
run-Lambda resultNo:在没有上下文对象的情况下调用
withthisLambda resultNo:以上下文对象作为参数。是顶层函数
applythisContext object(不会返回闭包结果)Yes
alsoitContext object(不会返回闭包结果)Yes
let

可用于对调用链的结果调用一个或多个函数。例如,下面的代码打印一个集合上的两个操作的结果:

val numbers = mutableListOf("one", "two", "three", "four", "five")
val resultList = numbers.map { it.length }.filter { it > 3 }
println(resultList) 

val numbers = mutableListOf("one", "two", "three", "four", "five")
numbers.map { it.length }.filter { it > 3 }.let { 
    println(it)
    // and more function calls if needed
} 

Let通常用于执行包含非空值的代码块。要对非空对象执行操作,请使用安全调用运算符?。

val str: String? = "Hello"   
//processNonNullString(str)       // compilation error: str can be null
val length = str?.let { 
    println("let() called on $it")        
    processNonNullString(it)      // OK: 'it' is not null inside '?.let { }'
    it.length
}

Fragment 中获取参数

arguments?.let {
   param1 = it.getString(ARG_PARAM1)
}
with

不能判空
建议在不需要使用返回结果时使用with来调用上下文对象上的函数。在代码中,with可读为“使用此对象,执行以下操作。”

val numbers = mutableListOf("one", "two", "three")
with(numbers) {
    println("'with' is called with argument $this")
    println("It contains $size elements")
}
run

Run的功能与with相同,但它是作为扩展函数实现的。
当你的lambda既初始化对象又计算返回值时,Run很有用。

val service = MultiportService("https://example.kotlinlang.org", 80)

val result = service.run {
    port = 8080
    query(prepareRequest() + " to port $port")
}

// the same code written with let() function:
val letResult = service.let {
    it.port = 8080
    it.query(it.prepareRequest() + " to port ${it.port}")
}

还可以将run作为非扩展函数调用。run的非扩展变体没有上下文对象,但它仍然返回lambda结果。非扩展运行允许您在需要表达式的地方执行由几个语句组成的块。

val hexNumberRegex = run {
    val digits = "0-9"
    val hexDigits = "A-Fa-f"
    val sign = "+-"

    Regex("[$sign]?[$digits$hexDigits]+")
}

for (match in hexNumberRegex.findAll("+123 -FFFF !%*& 88 XYZ")) {
    println(match.value)
}
apply

建议您将它用于不返回值且主要操作receiver对象成员的代码块。apply最常见的用例是用于对象配置。这样的调用可以理解为“对对象应用下列赋值”。

val adam = Person("Adam").apply {
    age = 32
    city = "London"        
}
println(adam)

apply的另一个用例是在多个调用链中包含apply,以进行更复杂的处理。

Fragment 的参数赋值

arguments = Bundle().apply {
      putString(ARG_PARAM1, param1)
}
also

对于需要引用对象而不是其属性和函数的操作,或者当您不想从外部作用域遮挡this引用时,使用also。

当您在代码中看到also时,您可以将其读为“并且还对对象执行以下操作”。

val numbers = mutableListOf("one", "two", "three")
numbers
    .also { println("The list elements before adding new one: $it") }
    .add("four")
资料

https://kotlinlang.org/docs/scope-functions.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值