针对空指针问题, Kotlin有专门的针对语法以避免问题。
- 可空类型
默认声明的变量是不能为 null 的,如果要使变量能为 null, 需要添加 ?
val name: String = null // 编译错误
var name = "harry"
name = null // 编译错误
// 正确的写法
var name: String? = null
var name: String? = "harry"
name = null
同样, 方法的返回值,如果可能返回空,也要加 ?
fun test(): String? = if ... else null
- 安全空访问
智能类型转换是一个相当好的特点,可以安全访问可空变量, 如:
var name: String? = "harry"
if (name != null) {
println(name.length)
}
但是,考虑到多个 null 检测,代码的可读性就变差,示例:
class Country(val name: String)
class City(val name: String, val country: Country?)
class Persion(val name: String, val city: City?)t
fun getCountryName(persion: Persion?): String? {
var countryName: String? = null
if (persion != null) {
val city = persion.city
if (city != null) {
val country = city.conutry
if (country != null) {
countryName = country.name
}
}
}
return countryName
}
嵌套的 null 检测, 代码可读性较差, Kotlin提供了简明的写法 ?.
fun getCountryName(persion: Persion?) = persion?.city?.country?.name
- 非空断言
有时候,要将可空的变量转成非空的, 常见于Java, 我们知道变量是不可能为 null 的, 可以用非空断言 !!
val nullableString: String? = "hello world"
val string: String = nullableString!!
var name: String? = null
name = "jason"
val len = name!!.length
- Elvis操作
Java里面, 有个三元操作?:
, 实现 if … else … 的赋值。Kotlin 也有这个操作符, 但是用法不一样,如:
val nullableName: String? = ...
val name: String = nullableName ?: "default_name"
如上代码意思了, 如果 nullableName 为 null, 就赋值 “default_name”。
?:
主要用于如果变量为 null就设置默认值。
- 安全类型转换
如果想安全地进行类型转换, 当转换失败时结果 null, 这种情况可用 as?
val location: Any = "London"
val safeString: String? = location as? String
val safeInt: Int? = location as? Int
参考
《Programming Kotlin》Stephen Samuel ,Stefan Bocutiu
《Kotlin in Action》Dmitry Jemerov,Svetlana Isakova