可空类型与非空类型
在 Kotlin 中,类型系统区分一个引用可以容纳 null
(可空引用)还是不能容纳(非空引用)。 例如,String
类型的常规变量不能容纳 null
:
var a: String = "abc"
a = null // 编译错误
复制代码
如果你调用 a 的方法或者访问它的属性,它保证不会导致 NPE,这样你就可以放心地使用:
val l = a.length
复制代码
如果要允许为空,我们可以声明一个变量为可空字符串,写作 String?
:
var b: String? = "abc"
b = null // ok
print(b)
复制代码
但是如果你想访问 b 的同一个属性,那么这是不安全的,并且编译器会报告一个错误:
val l = b.length // 编译错误:变量“b”可能为空
复制代码
安全的调用
- 首先,你可以显式检查 b 是否为
null
val l = if (b != null) b.length else -1
复制代码
- 你的第二个选择是安全调用操作符,写作
?.
:
val a = "Kotlin"
val b: String? = null
println(b?.length) //打印:null
println(a?.length) // 无需安全调用
复制代码
安全调用在链式调用中很有用。例如,如果一个员工 Bob 可能会(或者不会)分配给一个部门, 并且可能有另外一个员工是该部门的负责人,那么获取 Bob 所在部门负责人(如果有的话)的名字,我们写作:
bob?.department?.head?.name
复制代码
如果要只对非空值执行某个操作,安全调用操作符可以与 let 一起使用:
val listWithNulls: List<String?> = listOf("Kotlin", null)
for (item in listWithNulls) {
item?.let { println(it) } // 输出 Kotlin 并忽略 null
}
复制代码
安全调用也可以出现在赋值的左侧。这样,如果调用链中的任何一个接收者为空都会跳过赋值,而右侧的表达式根本不会求值:
// 如果 `person` 或者 `person.department` 其中之一为空,都不会调用该函数:
person?.department?.head = managersPool.getManager()
复制代码
摘自:kotlin语言中文站