简单说下Kotlin 作用域函数中 apply 和 also 为什么不能空安全调用?

在 Kotlin 中,applyalso 这两个作用域函数确实不能直接使用空安全调用操作符(?.),这有几个原因:

原因分析

1. 语法设计决定

applyalso 被设计为接收者对象的扩展函数,它们的签名如下:

kotlin 复制代码
inline fun <T> T.apply(block: T.() -> Unit): T
inline fun <T> T.also(block: (T) -> Unit): T

它们被定义为 T 的扩展函数,而不是 T? 的扩展函数。这意味着它们只能在非空对象上调用。

2. 设计哲学

这两个函数的核心目的是在已存在的对象上执行操作:

  • apply:配置对象并返回自身
  • also:执行副作用并返回自身

如果对象可能为空,那么就没有"自身"可以操作或返回。

解决方法

方法1:使用安全调用结合 let

kotlin 复制代码
val result: String? = nullableString?.let { 
    it.apply { 
        // 这里 it 是非空的
        toUpperCase()
    }
}

方法2:使用 run(更简洁)

kotlin 复制代码
val result = nullableString?.run {
    // 这里 this 是非空的
    toUpperCase()
    // 最后一行是返回值
}

方法3:使用 Elvis 操作符提供默认值

kotlin 复制代码
val result = (nullableString ?: "").apply {
    // 这里 this 是非空的(因为提供了默认值)
    toUpperCase()
}

各作用域函数的空安全特性对比

函数 是否支持 ?. 替代方案
apply ❌ 不支持 使用 ?.run { }
also ❌ 不支持 使用 ?.let { }
let ✅ 支持 obj?.let { }
run ✅ 支持 obj?.run { }
with ❌ 不支持 使用安全调用先判断
takeIf/takeUnless ✅ 支持 obj?.takeIf { }

实际使用建议

kotlin 复制代码
// 错误:无法编译
// nullableString?.apply { ... }

// 正确:使用 run 替代
nullableString?.run {
    println(length)  // 这里 this 是非空的
}

// 正确:使用 let 替代 also
nullableString?.let { str ->
    println(str.length)  // str 是非空的
    // 如果需要返回原对象,可以最后写 str
    str
}

// 正确:明确处理空值
nullableString?.also { 
    // 这里 it 是非空的
    println(it)
} ?: run {
    println("值为空")
}

总结来说,applyalso 不支持空安全调用是 Kotlin 语言设计的有意选择,目的是保持函数职责的清晰。当你需要空安全的作用域函数时,应该使用 letrun 配合安全调用操作符。

相关推荐
我命由我1234510 小时前
Android Jetpack Compose - Compose 重组、AlertDialog、LazyColumn、Column 与 Row
android·java·java-ee·kotlin·android studio·android jetpack·android-studio
愤怒的代码10 小时前
在 Android 中执行 View.invalidate() 方法后经历了什么
android·java·kotlin
PoppyBu12 小时前
Ubuntu20.04版本上安装最新版本的scrcpy工具
android·ubuntu
执念、坚持12 小时前
Property Service源码分析
android
用户416596736935512 小时前
在 ViewPager2 + Fragment 架构中玩转 Jetpack Compose
android
GoldenPlayer12 小时前
Gradle脚本执行
android
用户745890020795412 小时前
Android进程模型基础
android
we1less12 小时前
[audio] Audio debug
android
Jomurphys13 小时前
AndroidStudio - TOML
android
有位神秘人13 小时前
Android最新动态权限申请工具
android