系列文章的知识点会以《Kotlin实战》这本书中顺序编写,在将书中知识点展示出来同时,我也会添加对应的Java代码用于对比学习和更好的理解。
[Kotlin教程(五)类型]
[Kotlin教程(六)Lambda]
这一章实际上在《Kotlin实战》中是第六章,在Lambda之后,但是这一章的内容实际上是Kotlin的一大特色之一。因此,我将此章的内容提到了前面汇总。
可空性
可空性是Kotlin类型系统中帮助你避免NullPointerException错误的特性。
可空类型
如果一个变量可能为null,对变量的方法的调用就是不安全的,因为这样会导致NullPointerException。例如这样一个Java函数:
int strLen(String s) {
return s.length();
}
如果这个函数被调用的时候,传给它的是一个null实参,它就会抛出NullPointerException。那么你是否需要在方法中增加对null的检查呢?这取决与你是否期望这个函数被调用的时候传给它的实参可以为null。如果不可以的话,我们用Kotlin可以这样定义:
fun strLen(s: String) = s.length
看上去与Java没有区别,但是你尝试调用strLen(null)
就会发现在编译期就会被标记成错误。因为在Kotlin中String
只能表示字符串,而不能表示null,如果你想支持这个方法可以传null,则需要在类型后面加上?
:
fun strLen(s: String?) = if(s != null) s.length else 0
?
可以加在任何类型的后面来表示这个类型的变量可以存储null引用:String?
、 Int?
、MyCustomType?
等。
一旦你有一个可空类型的值,能对它进行的操作也会受到限制。例如不能直接调用它的方法:
val s: String? = ""
// s.length //错误,only safe(?.) or non-null asserted (!!.) calls are allowed
s?.length //表示如果s不为null则调用length属性
s!!.length //表示断言s不为null,直接调用length属性,如果s运行时为null,则同样会crash
也不能把它赋值给非空类型的变量:
val x: String? = null
// val y: String = x //Type mismatch
也就是说,加?
和不加可以看做是两种类型,只有与null进行比较后,编译器才会智能转换这个类型。
fun strLen(s: String?) = if(s != null) s.length else 0
这个例子就与null进行比较,于是String?
类型被智能转换成String
类型,所以可以直接获取length属性。
Java有一些帮助解决NullPointerException问题的工具。比如,有些人会使用注解(@Nullable和@NotNull)来表达值得可空性。有些工具可以利用这些注解来发现可能抛出NullPointerException的位置,但这些工具不是标