参考 https://mp.weixin.qq.com/s/VkJHHfhU8T8fz5sY9Rpyyw
看了kotlin对null的处理之后,分分钟想甩掉java啊!
可空类型与非空类型
Kotlin 的类型系统旨在消除来自代码空引用的危险,许多编程语言(包括 Java)中最常见的陷阱之一是访问空引用的成员,导致空引用异常。在 Java 中, 就是我们熟悉的 NullPointerException ,或简称 NPE。
要知道,在大多数项目的 crash 中,NPE 比率占到了半数以上,毫不夸张。
但是,Kotlin 的类型系统也只能说尽可能避免 NullPointerException,并不是使用 Kotlin 就再也没有 NPE。如果使用 Kotlin 时存在 NPE,可能的原因是:
1.显式调用 throw NullPointerException();
2.使用了下文描述的 !! 操作符;
3.外部 Java 代码导致的;
4.对于初始化,有一些数据不一致(如一个未初始化的 this 用于构造函数的某个地方)。
在 Kotlin 中,其类型系统严格区分一个引用可以容纳 null (可空引用)还是不能容纳(非空引用)。也就是说,一个变量是否可空必须显示声明,对于可空变量,在访问其成员时必须做空处理,否则无法编译通过。
空处理
val bLength: Int? = b?.length
如果 b 非空,就返回 b.length;否则返回 null。
//如此优雅...
bob?.department?.head?.name
安全调用在链式调用中很有用。例如,如果一个员工 Bob 可能会(或者不会)分配给一个部门, 并且可能有另外一个员工是该部门的负责人,那么获取 Bob 所在部门负责人(如果有的话)的名字
如果任意一个属性(环节)为空,这个链式调用就会返回 null,而不是出现 NPE。
Elvis 操作符
val bLength: Int = if (b != null) b.length else -1
val bLength = b?.length ?: -1
fun foo(node: Node): String? {
val parent = node.getParent() ?: return null
val name = node.getName() ?: throw IllegalArgumentException("name expected")
// ……
}
如果 ?: 左侧表达式非空,elvis 操作符就调用其左侧表达式,否则调用右侧表达式。
!! 操作符
val bLength = b!!.length
如果 b 非空则返回 b,否则就会抛出一个 NPE 异常:
安全的类型转换
val aInt: Int? = a as? Int
等号左侧声明一个可为空的 Int 常量接收右侧的值,而右侧表示:如果 a 是 Int 类型,则返回 a,否则返回 null。
可空类型的集合
val nullableList: List<Int?> = listOf(1, 2, null, 4)
val intList: List<Int> = nullableList.filterNotNull()
如果你有一个可空类型元素的集合,并且想要过滤非空元素,你可以使用 filterNotNull 来实现。