欢迎关注 二师兄Kotlin
转载请注明出处 二师兄kotlin
kotlin "null"
Kotlin的类型设计系统旨在消灭NullPinterException(下文简称NPE),从此妈妈再也不用担心代码里面的空指针了。
当然,Kotlin只是在很大程度上帮你干掉NPE,如果你非得搞出来,Kotlin也满足你的欲望,以下几种情况能帮你看到熟悉的NPE:
- 显示调用
throw NullPointerException()
- 使用
!!
操作符 - 外部java代码导致*
- 一些在构造函数中未初始化的变量到处使用
在Kotlin里,系统可以区分引用是否可以null 引用,举栗来说,一个String
的常规变量不能持有null
:
var a: String = "abc"
a = null // compilation error
为了允许它为null,我们需要声明一个允许为null的string,写法为String?
:
var b: String? = "abc"
b = null // ok
现在,如果你调用 a
的一个方法或者访问属性, 可以打一千个保票不会引起 NPE
:
val l = a.length
但是如果你访问 b
的一些属性, 编译器会直接给你一个error:
val l = b.length // error: variable 'b' can be null
但是就算你给我报error,我还是想访问啊,肿么办。。。
解决方法如下:
方法1 自己做检查
首先,我们可以显示的检查 b
是否为空,手动处理不同的情况:
val l: Int = if (b != null) b.length else -1
或者
if (b != null && b.length > 0)
print("String of length ${b.length}")
else
print("Empty string")
方法2 安全调用操作符
Kotlin 新增了一个安全调用操作符 ?.
:
b?.length
如果当b
不为null时候就会返回b.length
,否则返回null
。
安全调用也适用于链式调用,比如:
bob?.department?.head?.name
其中的任何一个对象为null就会返回null
。
如果你确定你要执行的操作只针对非空对象,那么你可以?.
配合let
一起使用:
val listWithNulls: List<String?> = listOf("A", null)
for (item in listWithNulls) {
item?.let { println(it) } // prints A and ignores null
}
方法3 艾维斯操作符(名字有木有觉得略屌)
首先是普通青年的写法:
val l: Int = if (b != null) b.length else -1
文艺青年Kotlin写法之艾维斯操作符
-----?:
:
val l = b?.length ?: -1
这个操作符跟&&
类似,操作符左侧满足要求,就不再执行右侧.
前方高能~~~~
这个操作符能直接把 throw
和 return
当作表达式放在右侧:
fun foo(node: Node): String? {
val parent = node.getParent() ?: return null
val name = node.getName() ?: throw IllegalArgumentException("name expected")
// ...
}
方法4 !! 操作符
val l = b!!.length
要么返回b的长度length
,要么就一言不合抛出NPE
Kotlin之类型转换
java中类型强转一失败就抛异常 ClassCastException
, Kotlin友好多了,转不成功给你个 null
:
val aInt: Int? = a as? Int
本节结尾总结一下:
Kotlin强大的操作符设计屏蔽了java最大的坑--NPE,而且整体使用上感觉非常舒服,建议java的小伙伴们都来试一试。