在Kotlin中,?:操作符被称为Elvis操作符,它用于处理可空类型的值,提供一种简洁的方式来指定当左侧表达式为null时的默认值。
Elvis操作符的基本语法和执行逻辑是: 表达式 A ?: B 的执行逻辑如下:如果表达式 A 的值不为 null,则整个表达式的值就是 A;如果 A 为 null,则返回 B 的值。右侧表达式 B 仅在左侧为 null 时才会求值,这类似于短路求值。例如:
kotlin
var age: String? = null
val result = age?.toInt() ?: 0 // 如果 age 为 null,result 为 0
println(result) // 输出 0
它常与安全调用符(?.)结合使用,以避免NullPointerException,并提供默认值。常见场景包括:
- 变量空安全赋值:例如
val username = user?.name ?: "Guest",当user或user.name为null时返回"Guest"。 - 函数返回值处理:例如
fun getTextLength(text: String?): Int { return text?.length ?: 0 },确保返回非负整数。 - 链式调用保护:例如
val city = order?.customer?.address?.city ?: "Unknown City",防止属性访问链中的null。 - 配合异常抛出:例如
val address = person.company?.address ?: throw IllegalArgumentException("No Address"),在缺少数据时抛出有意义的异常。
与相关操作符对比: Elvis操作符与安全调用符(?.)和非空断言(!!)有明显区别:
?.:安全调用,如果左侧为null则返回null,不继续执行。?::空合并,提供默认值替代null。!!:强制解包,如果左侧为null则抛出NullPointerException。
进阶技巧和注意事项包括: 右侧表达式可以是任何合法表达式,包括函数调用或复杂逻辑,且支持延迟计算。例如:
kotlin
val data = cache.get(key) ?: { println("缓存未命中"); database.query(key) }()
类型推断方面,编译器会自动推断两侧类型的共同父类型。例如 val num: Number = intValue ?: 0.0 中,Number 是 Int 和 Double 的共同父类型。使用时需注意避免在右侧表达式中引入不必要的副作用,并合理使用括号处理优先级问题。