? 可空类型
kotlin和Java的类型系统之间的一个很重要的区别就是,Kotlin对可空类型的显示支持
也就是说你可以声明一个变量,并且使用可空类型?来表示这个变量是可以为null的
比如:
java:
int StrLen(String s){return s.length} //这个函数并不安全,原因是传入的参数s如果是null,就会报空指针异常
kotlin:
fun StrLen(s:String?):Int = s.length //不能直接调用length方法
//1、这里使用了可空类型?,?可以加载任何类型的后面来表示这个类型的变量可以为null //2、可空类型的变量在使用的时候不能直接调用它的方法
//3、也不能把可空类型的值传给非空类型 /** *如 val x:String?=null var y:String =
x//把可空类型的x赋值给非空类型的y会报错:Type mismatch *同样也不能把一个可控类型的值传给拥有非空类型参数的函数如StrLen(x) 传给
StrLen(x:String) **/
?. 安全调用运算符
fun StrLen(s:String?):Int = s.length //不能直接调用length方法
//如果增加了null检查以后,就可以直接调用s.length了,如下: fun StrLen(s:String?):Int = if(s!=null)
s.lengthelse 0 //但是如果每个可空类型都这样检查会显得特别累赘,此时就用到了安全调用运算符?. s?.length 就相当于 if(s!=
null) s.length else null //如果s不为空就执行方法,如果为空就返回null
?: Elvis运算符(null合并运算符)
使用?:运算符可以设置当检查结果为空的时候的返回值
fun foo(s:String?){ val t:String = s ?: "" //如果?:左边的值不为空返回左边的值,如果为空返回"" }
//可以这样使用 a?. peroson?. name ?: "UnKnown" //如果?:左边为空则返回"UnKnown" //和throw运算符同事使用
//如果不为空就返回name,如果为空就抛出一个有意义的错误,而不是NullPointException val name = a?.person?.name
?:throw illegalargumentexception("UnKnown name")
//如果name为空就会报自定义的异常,防止下面代码调用而直接报空指针异常 println(name.length)//如果name为空就会报空指针异常
as? 安全转换运算符
尝试把值转换成给定的类型,如果类型不合适就返回null
foo as? Type -> foo is Type retrun (foo as Type) -> foo !is Type return null
//as?和?:联合使用 object as? Person ?: "not human" object as? Person ?: return false
!! 非空断言 (Kotlin不推荐使用非空断言,通常我们会用?:来防止程序运行时报空指针异常而崩溃)
如果值为null就抛出NullPointerException空指针异常
var s:String = s!! //如果s为null则会抛出空指针异常,并且异常会指向使用!!的这一行 println(s)
//如果s为null则会抛出空指针异常 //使用断!!可以很方便的在抛出空指针异常的时候定位到异常的变量的位置 //但是千万不要连续使用断言!!
//student!!.person!!.name//如果报空指针异常了则无法判断到底是student为空还是person为空,所以不要连续使用断言!!
let函数和?.同时使用来处理可空表达式
let函数只有左侧表达式不为空的时候才会调用
person?.let{ //内部的it一定是非空的 } //如果person为空就不会调用let函数
//如果person不为空才会调用let函数,所以?.和let函数配合使用可以很方便的处理可空表达式 person?.let {
println(it.name)//输出name}
以上就是简单的总结
参考资料:《Kotlin实战》
热门工具 换一换