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

199 阅读2分钟

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

原因分析

1. 语法设计决定

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

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

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

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

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

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

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

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

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

实际使用建议

// 错误:无法编译
// 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 配合安全调用操作符。