在 Kotlin 中,apply 和 also 这两个作用域函数确实不能直接使用空安全调用操作符(?.),这有几个原因:
原因分析
1. 语法设计决定
apply 和 also 被设计为接收者对象的扩展函数,它们的签名如下:
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("值为空")
}
总结来说,apply 和 also 不支持空安全调用是 Kotlin 语言设计的有意选择,目的是保持函数职责的清晰。当你需要空安全的作用域函数时,应该使用 let 或 run 配合安全调用操作符。