简单说下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 配合安全调用操作符。

相关推荐
BoomHe1 天前
Android AOSP13 原生 Launcher3 壁纸获取方式
android
Digitally1 天前
如何将联系人从 Android 转移到 Android
android
一直在想名1 天前
Flutter 框架跨平台鸿蒙开发 - 黑白屏
flutter·华为·kotlin·harmonyos
李小枫1 天前
webflux接收application/x-www-form-urlencoded参数
android·java·开发语言
爱丽_1 天前
MySQL `EXPLAIN`:看懂执行计划、判断索引是否生效与排错套路
android·数据库·mysql
NPE~1 天前
[App逆向]环境搭建下篇 — — 逆向源码+hook实战
android·javascript·python·教程·逆向·hook·逆向分析
yewq-cn1 天前
AOSP 下载
android
cch89181 天前
Laravel vs ThinkPHP:PHP框架终极对决
android·php·laravel
米码收割机1 天前
【Android】基于安卓app的汽车租赁管理系统(源码+部署方式+论文)[独一无二]
android·汽车
流星雨在线1 天前
安卓使用 Startup 管理三方 SDK 初始化
android·startup