Kotlin(1)之空安全

写在前面: Kotlin 在上半年的 Google 的 IO 大会上,被宣布成为 Android 开发的官方语言。当时乘着热度学习了一段时间,后来因为工作上的原因中断了Kotlin的学习。而近期 Kotlin 1.2 发布,正好现在又有时间学习,所以再次开始学习Kotlin,并且将学习过程中遇到的问题或者觉得需要记录的知识点以博客的形式记录下来,方便以后查阅。

正文:
在使用 java 语言进行开发时,永远避不过去的一个问题就是NPE(NullPointException)。在开发过程中需要对可能出现空指针异常的地方进行处理,稍不留神就可能出现。下面来看看在 Kotlin 中是如何解决这个问题的。
Kotlin 在设计之初就要消除 NullPointerException 。而如果在程序中出现了 NPE ,只可能是出现了一下情况:
- * 显式调用 throw NullPointerException() *
- * 使用了 !! 操作符 *
- * 外部 Java 代码导致的 *
- * 对于初始化,有一些数据不一致(如一个未初始化的this’用于构造函数的某个地方)。 *
在 Kotlin 中,类型系统区分一个引用可以容纳 null(可空引用)还是不能容纳(非空引用)。例如,String 类型的常亮变量不能容纳 null:

var a: String = "abc"
a = null // 编译错误

如果要允许为空,可以声明一个变量为可空字符串,写作String? :

var b: String? = "abc"
b = null // ok

现在,如果调用 a 的方法或者访问它的属性,它保证不会导致 NPE,这样就可以放心的使用:

val l = a.length

但是如果想要访问 b 的同一个属性就不行了,因为这是不安全的,所以无法编译成功:

val l = b.length//错误:变量"b"可能为空

但是如果还是需要访问该属性,有以下几种方式可以做到。

在条件中检查 null

val l = if (b != null) b.length else -1

编译器会跟踪所执行检查的信息,并允许你在 if 内部调动 length 。同时,也支持更复杂(更智能)的条件:

if (b != null && b.length >0) {
    print("String of length ${b.length}")
} else {
    print("Empty string")
}

请注意,这只适用于b是不可变的情况(即在检查和使用之间没有修改过的局部变量,或者不可覆盖并且有幕后字段的 val 成员),因为否则可能会发生在检查之后 b 又变成 null 的情况。

安全的调用
第二个选择是安全调用操作符,写作 ?. :

b?.length

如果 b 非空,就返回 b.length ,否则返回 null ,这个表达式的类型是 Int? 。
安全调用在链式调用中很有用。例如,如果一个员工 Bob 可能会(或者不会)分配给一个部门,并且可能有另外一个员工是该部门的负责人,那么获取 Bob所在部门负责人(如果有的话)的名字,可以写作:

bob?.department?.head?.name

如果任意一个属性(环节)为空,这个链式调用就会返回 null 。
如果要只对非空值执行某个操作,安全调用操作符可以与 let 一起使用:

val listWithNUlls: List<String?> = listof("A",null)
for (item in listWidthNulls) {
    item?.let{ 
        println(it) //如果item不为空,将item输出
    }
}

Elvis操作符
当我们有一个可空的引用r时,我们可以说“如果 r 非空,我使用它;否则使用某个非空的值x”:

val l : Int = if (b != null) b.length else -1

除了完整的 if 表达式,这还可以通过 Elvis操作符表达,写作?::

val l = b?.length ?: -1

如果 ?: 左侧表达式非空, elvis 操作符就返回其左侧表达式,否则返回右侧表达式。请注意,当且仅当左侧为空时,才会对右侧表达式求值。
请注意,因为 throw 和 return 在 Kotlin 中都是表达式,所以他们也可以在 elvis 操作符右侧。这可能会非常方便,例如,检查函数参数:

fun foo(node: Node): String?{
    val parent  = node.getParent() ?: return null
    val name = node.getName() ?: throw IllegalArgumentException("name expected")
}

!! 操作符
第三种选择是为NPE爱好者准备的:非空断言运算符(!!)将任何值转换为非空类型,若该值为空则抛出异常。我们可以写 b!! ,这会返回一个非空的 b 值或者如果 b 为空,就会抛出一个 NPE 异常:

val l = b!!.length

因此,如果你想要一个 NPE ,你可以得到它,但是必须显试要求它,否则不会不期而至。

安全的类型转换
如果对象不是目标类型,那么常规类型转换可能会导致 ClassCastException 。另一个选择是使用安全的类型转换,如果尝试转换不成功则返回 null :

val aInt: Int? = a as? Int

可空类型的集合
如果你有一个可空类型元素的集合,并且想要过滤非空元素,你可使用 filterNotNull 来实现:

val nullableList: List<Int?> = listOf(1,2,null,4)
val intList: List<Int> = nullableList.filterNotNull()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值