背景
既然Google官方已经宣布Kotlin为Android官方语言,我觉得,那就没什么可争议的,作为Android开发者必须开始学习,跟上节奏,相信不会很费力气。我学习的大部分知识来自Kotlin 学习之路,这哥们写得非常细心,真心赞。
Kotlin如何减少空指针异常
首先科普一下什么情况会导致NPE(空指针异常):
String str; //声明变量
//str="Hello Word"; //没有初始化变量
str.hashCode(); //报NPE
Kotlin 解决空安全,不是消灭 NPE,而是 允许 NPE 存在,但只会在它该出现的地方出现 。
可空类型与非空类型
所有数据类型默认都是非空(non-null)的,如果想把 null 赋给它们的对象,需要在类型后加上 ? ,声明为可空类型。Kotlin 这样设计有什么好处呢?我们只要将变量声明为非空类型,就可以放心地调用它的实例方法和实例变量,不用担心出现 NPE。
val str:String="Hello World"; //不可空类型
val str:String?=null; //可空类型
如何安全地调用可空类型变量的方法
上面讲到我们可以放心地调用非空类型变量的方法,可对于可空类型呢?要用到以下方法:
val str:String?=null; //可空类型
//str="Hello World";
str.?hashCode();
如果 ?. 前的对象不为 null,则调用 ?. 后的方法或属性,否则返回 null。
Elvis 操作符 ?:
比如这段 Java 代码:
File[] files = new File("C:\\Documents").listFiles();
System.out.println((files != null) ? files.length : "Empty");
如果 files 数组为 null,则打印 "Empty",如果不为 null,则打印数组的长度。
用 Kotlin 写是这样的:
val files = File("C:\\Documents").listFiles()
println(files?.size ?: "Empty")
可以这样理解 Kotlin 里的 Elvis 操作符:
A ?: B 等价于 if(A == null) B
A?.B ?: C 等价于 if(A != null) A.B else C
let 函数
Java写法:
File[] files = (new File("C:\\Documents")).listFiles();
if(files != null) {
for(int i = 0; i < files.length; i++) {
File file = files[i];
System.out.println(file.getName());
}
}
Kotlin写法:
val files = File("C:\\Documents").listFiles()
// files 的类型自动推断为 Array<File>?
files?.let {
for (file in files)
println(file.name)
}
相比上面的.?,这个?.let只是多了个括号。
非空操作符 !!
Kotlin写法:
val files = File("C:\\Documents").listFiles()!! // files 的类型自动推断为 Array<File>
for(file in files)
println(file.name)
Java写法:
File[] var1 = (new File("C:\\Documents")).listFiles();
if(var1 == null)
Intrinsics.throwNpe(); // 抛出 NPE
File[] files = var1;
也就是说加了!!的地方,如果为null,则会报NPE,所以说Kotlin还是存在空指针异常的,只不过Kotlin官方不建议代码中有太多的!!,否则就浪费了Kotlin设计良好的非空类型。
总结
我把一个用Java写的项目全改成Kotlin了,发觉原来的项目里存在太多会出现NPE的地方,而这些地方往往在我们写代码、测试时是无法发现的,只有在生产环境下,看后台的log才发现这些问题,所以Kotlin这个设计还是非常棒的!
写着写着,心里有些纠结,本着自己消化总结后写一篇通俗易懂的,可是后来发现原文写得太好了,纠结纠结,再次感谢作者,附上原文链接吧!