Kotlin 作用域函数之let、with、run、also、apply的使用笔记

简介: `Kotlin` 标准库包含几个函数,目的是在对象的上下文中执行代码块。**当对一个对象调用这样的函数并提供一个 `lambda` 表达式时,会形成一个临时作用域。在此作用域中,可以访问该对象而无需其名称。这些函数称为作用域函数**。共有以下五种:`let、run、with、apply 以及 also`。

作用域函数

Kotlin 标准库包含几个函数,目的是在对象的上下文中执行代码块。当对一个对象调用这样的函数并提供一个 lambda 表达式时,会形成一个临时作用域。在此作用域中,可以访问该对象而无需其名称。这些函数称为作用域函数。共有以下五种:let、run、with、apply 以及 also

作用域函数区别

  • 引用上下文对象的方式
引用上下文对象的方式 作用域函数
it letalso 可以用it ,主要用做lambda表达式中的参数
this runwith apply 通过this引用上下文对象,当访问类中的变量或方法时,可以使用this,也可以省略
  • 函数返回值
返回值 作用域函数
返回上下文对象 apply also返回上下文对象
返回表达式结果 letrunwith 返回lambda表达式结果,其中表达式也可以没有结果

let

let常用的两个作用:一是定义对象在特定的作用域范围内,另一个更常用的场景是对对象执行 xxx ?.let{ },在 lambda 表达式中执行操作,以一种更优雅的形式来判空。

 fun letFunc() {
    //let{}中可以使用it来代表intList对象
    val intList: List<Int> = arrayListOf(1, 2, 3)
    intList.let {
        println(it.size) //3
        println(it.lastIndex) //2
    }

    //通过?.let{}进行判断,如何为空,后面就不会再执行
    val list2: List<Int>? = null
    list2?.let {
        println(it.size)
    }
 }

with

 fun withExam(person: Person = Person("xmkp", 18, 1)) {
     //将person对象以参数的方式传递到Lambda表达式中
     with(person) {
         println(name)
         println(age)
         println(sex)
     }
 }
 
data class Person(var name: String, var age: Int, var sex: Int)

非扩展函数with(){}的使用,一般不需要用返回值, 而是将上下文对象作为参数传递到Lambda表达式中,典型应用是对一个对象在Lambda表达式中调用它的多个函数时使用,如with(RecyclerView.Holder){}

run

runwith的作用基本一致,run可以以扩展函数的方式调用,通用场景:当表达式中同时包含对象初始化及返回值计算时使用

 fun runExam() {
     val list = arrayListOf(1, 2, 3)
     val size = list.run {
         list.add(4) //添加一条数据
         list.size  //返回最终的size
     }
     println("list.size:$size") //执行结果:list.size:4
 }

also

also通常执行将上下文对象作为参数的操作,返回值是上下文对象本身。lambda表达式中用it来表示上下文对象,可以将also理解为并且用该对象执行以下操作

 fun alsoExam() {
     val list = arrayListOf(1, 2, 3)
     list.also {
         println("添加前:$list")
     }.add(4)
     println("添加后:$list")
 }
    
 //执行结果:
 // 添加前:[1, 2, 3]
 // 添加后:[1, 2, 3, 4]

apply

lambda表达式中不返回值,且主要是对对象成员进行操作的场景使用apply典型使用场景:对象的配置,可以将apply理解为将以下赋值应用于对象

 fun applyExam() {
     val person = Person().apply {
         //给Person对象设置属性
         name = "xxx"
         age = 30
         sex = 1
     }
     println(person) // 执行结果:Person(name=xxx, age=30, sex=1)
 }

总结

作用域函数 上下文对象 返回值 使用场景
let it 表达式结果,也可不返回 1、定义对象在特定的作用域范围内;
2、对象执行 xxx ?.let{ }来判空
with this 表达式结果,也可不返回 对一个对象在Lambda表达式中调用它的多个函数时使用,如with(RecyclerView.Holder){}
run this 表达式结果,也可不返回 当表达式中同时包含对象初始化及返回值计算时使用
also it 上下文对象 附加效果,相当于并且对该对象执行以下操作
apply this 上下文对象 lambda表达式中执行对象的配置,可以理解为**将以下赋值应用于对象

takeIf 与 takeUnless

takeIf、takeUnless 函数可以以链式调用的方式进行对象状态检查,是单个对象的过滤函数。个人认为可以简单理解成是对if/else的链式调用。使用规则:

  • takeIf: 如果takeIf后面的表达式或闭包符合要求,则takeIf返回此对象;否则返回null
  • takeUnless:与takeIf用法相反,如果takeUnless不匹配后面的表达式或闭包,则返回该对象;否则返回null

使用举例:

 fun takeXXFunc() {
     val num = Random.nextInt(100)
     val evenOrNull = num.takeIf { it % 2 == 0 } //结果为偶数 或 null
     val oddOrNull = num.takeUnless { it % 2 == 0 } //结果为奇数 或 null
     print("evenOrNull:$evenOrNull, oddOrNull:$oddOrNull")
    }

执行结果:

//奇数:evenOrNull:null, oddOrNull:69
//偶数:evenOrNull:4, oddOrNull:null

takeIf结合let使用

 /**
  * takeIf结合let使用
  */
 fun takeIfExam() {
     val list = arrayListOf(1, 2, 3)
     list.takeIf { it.size < 4 }?.let {
         println("list:$list")
     }
 }

执行结果:

list:[1, 2, 3]

如果改为val list = arrayListOf(1, 2, 3, 4),则takeIf判断不成立,直接返回null,后面就不会再执行了。

相关文章
|
2月前
|
Java 调度 Android开发
Android经典实战之Kotlin的delay函数和Java中的Thread.sleep有什么不同?
本文介绍了 Kotlin 中的 `delay` 函数与 Java 中 `Thread.sleep` 方法的区别。两者均可暂停代码执行,但 `delay` 适用于协程,非阻塞且高效;`Thread.sleep` 则阻塞当前线程。理解这些差异有助于提高程序效率与可读性。
51 1
|
2月前
|
Java Android开发 开发者
Kotlin 循环与函数详解:高效编程指南
高效编程实践 • 避免不必要的循环 - 尽量使用集合操作如 map、filter 来减少显式的循环。 • 使用尾递归优化 - 对于需要大量递归的情况,考虑使用尾递归以优化性能。 • 内联函数 - 对于传递 Lambda 表达式的函数,使用 inline 关键字可以减少运行时开销。 通过上述指南,您应该能够更好地理解 Kotlin 中的循环和函数,并能够编写更加高效和简洁的代码。Kotlin 的设计哲学鼓励开发者编写易于理解和维护的代码,而掌握循环和函数是实现这一目标的关键步骤。 如果您想了解更多关于 Kotlin 的循环和函数的信息,以下是一些官方文档和资源,它们可以提供额外的参考
35 1
|
2月前
|
Java Kotlin
Kotlin 循环与函数详解:高效编程指南
Kotlin中的循环结构让你能轻松遍历数组或范围内的元素。使用`for`循环结合`in`操作符,可以简洁地访问数组中的每个项,如字符串数组或整数数组。对于范围,可以用`..`来定义一系列连续的值并进行迭代。此外,Kotlin支持通过`break`和`continue`控制循环流程。函数则允许封装可复用的代码块,你可以定义接受参数并返回值的函数,利用简写语法使代码更加紧凑。例如,`myFunction(x: Int, y: Int) = x + y`简洁地定义了一个计算两数之和的函数。
42 1
|
3月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin中常见作用域函数
**Kotlin作用域函数概览**: `let`, `run`, `with`, `apply`, `also`. `let`安全调用并返回结果; `run`在上下文中执行代码并返回结果; `with`执行代码块,返回结果; `apply`配置对象后返回自身; `also`附加操作后返回自身
42 8
|
3月前
|
Swift iOS开发 Kotlin
苹果iOS新手开发之Swift中实现类似Kotlin的作用域函数
Swift可通过扩展实现类似Kotlin作用域函数效果。如自定义`let`, `run`, `with`, `apply`, `also`,增强代码可读性和简洁性。虽无直接内置支持,但利用Swift特性可达成相似功能。
51 7
|
Kotlin
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(二)
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(二)
120 0
|
Kotlin
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(一)
【Kotlin】Kotlin 构造函数 ( 主构造函数 | 主构造函数声明属性 | init 初始化代码块 | 次构造函数 | 构造函数委托 | 调用构造函数创建实例对象 )(一)
273 0
|
22天前
|
Android开发 开发者 Kotlin
告别AsyncTask:一招教你用Kotlin协程重构Android应用,流畅度飙升的秘密武器
【9月更文挑战第13天】随着Android应用复杂度的增加,有效管理异步任务成为关键。Kotlin协程提供了一种优雅的并发操作处理方式,使异步编程更简单直观。本文通过具体示例介绍如何使用Kotlin协程优化Android应用性能,包括网络数据加载和UI更新。首先需在`build.gradle`中添加coroutines依赖。接着,通过定义挂起函数执行网络请求,并在`ViewModel`中使用`viewModelScope`启动协程,结合`Dispatchers.Main`更新UI,避免内存泄漏。使用协程不仅简化代码,还提升了程序健壮性。
45 1
|
2月前
|
调度 Android开发 开发者
【颠覆传统!】Kotlin协程魔法:解锁Android应用极速体验,带你领略多线程优化的无限魅力!
【8月更文挑战第12天】多线程对现代Android应用至关重要,能显著提升性能与体验。本文探讨Kotlin中的高效多线程实践。首先,理解主线程(UI线程)的角色,避免阻塞它。Kotlin协程作为轻量级线程,简化异步编程。示例展示了如何使用`kotlinx.coroutines`库创建协程,执行后台任务而不影响UI。此外,通过协程与Retrofit结合,实现了网络数据的异步加载,并安全地更新UI。协程不仅提高代码可读性,还能确保程序高效运行,不阻塞主线程,是构建高性能Android应用的关键。
41 4
|
3月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
**Kotlin中的`by lazy`和`lateinit`都是延迟初始化技术。`by lazy`用于只读属性,线程安全,首次访问时初始化;`lateinit`用于可变属性,需手动初始化,非线程安全。`by lazy`支持线程安全模式选择,而`lateinit`适用于构造函数后初始化。选择依赖于属性特性和使用场景。**
107 5
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?