空安全
可空类型和非空类型
在 Kotlin 类型系统中可以为空和不可为空的引用是不同的。比如,普通的 String 类型的变量不能为空:
var a: String ="abc"
a = null //编译错误
允许为空,我们必须把它声明为可空的变量:
var b: String? = "abc"
b = null
现在你可以调用 a 的方法,而不用担心 NPE 异常了:
val l = a.length()
但如果你想使用 b 调用同样的方法就有可能报错了:
val l = b.length() //错误:b 不可为空
在条件中检查 null
首先,你可以检查 b 是否为空,并且分开处理下面选项:
val l = if (b != null)
b.length()
else
-1
编译器会跟踪你检查的信息并允许在 if 中调用 length()。更复杂的条件也是可以的:
if (b != null && b.length() >0)
print("Stirng of length ${b.length}")
else
print("Empty string")
注意只有在 b 是不可变时才可以
安全调用
第二个选择就是使用安全操作符,?.:
b?.length()
如果 b 不为空则返回长度,否则返回空。这个表达式的的类型是 Int?
安全调用在链式调用是是很有用的。比如,如果 Bob 是一个雇员可能分配部门(也可能不分配),如果我们想获取 Bob 的部门名作为名字的前缀,就可以这样做:
bob?.department?.head?.name
这样的调用链在任何一个属性为空都会返回空。
Elvis 操作符
当我们有一个 r 的可空引用时,我们可以说如果 r 不空则使用它,否则使用使用非空的 x :
val l: Int = if (b != null) b.length() else -1
尽管使用 if 表达式我们也可以使用 Elvis 操作符,?:
val l = b.length()?: -1
如果 ?: 左边表达式不为空则返回,否则返回右边的表达式。注意右边的表带式只有在左边表达式为空是才会执行
注意在 Kotlin 中 throw return 是表达式,所以它们也可以在 Elvis 操作符右边。这是非常有用的,比如检查函数参数是否为空;
fun foo(node: Node): String? {
val parent = node.getParent() ?: return null
val name = node.getName() ?: throw IllegalArgumentException("name expected")
//...
}
!! 操作符
第三个选择是 NPE-lovers。我们可以用 b!! ,这会返回一个非空的 b 或者抛出一个 b 为空的 NPE
val l = b !!.length()
安全转换
普通的转换可能产生 ClassCastException 异常。另一个选择就是使用安全转换,如果不成功就返回空:
val aInt: Int? = a as? Int
网友评论