16.空指针检测
kt默认所有参数和变量都不能为空,编译期就进行了空指针检测,若尝试传空则报错,ef:
可空类型系统:类型后加?,针对非得传空的情况,ef:
此时调用方法的参数报错,做个非空判断即可,ef:
fun doStudy(study: Study?) {
if (study != null) {
study.readBooks()
study.doHomework()
}
}
//显然以上繁琐,于是出现判空辅助工具
判空辅助工具
?.操作符(被修饰的对象 若非空则执行方法,若为空则不执行),ef:
fun doStudy(study: Study?) {
study?.readBooks()//若study非空才执行readBooks
study?.doHomework()//若study非空才执行doHomework
}
?:操作符(操作符左右两边都接收一个表达式,若左边非空则输出结果为左边,否则为右边),ef:
val c = if (a ! = null) {
a
} else {
b
}
//用?:操作符,简化后,若a非空则结果为a,否则为b
val c = a ?: b
//以下再举例一个,编写一个函数获取一段文本长度
fun getTextLength(text : String?) : Int {
if(text != null){
return text.length
}
return 0
}
//结合之前的?.和?:操作符后,并由kt的类型推导机制,简化如下
fun getTextLength(text : String?) = text?.length ?: 0
非空判断工具
!!操作符(被修饰的对象 用于强调非空,强行通过编译)
有时kt的空指针检测不是很智能,比如有时候我们已在逻辑上处理了空指针,但kt不知,所以还是会报错,ef:
var content: String? = "hello"//全局content
fun main() {
//此处已做了非空判断
if (content != null) {
printUpperCase()
}
}
fun printUpperCase() {
//但kt不那么智能,它不知道在该方法外已做了非空处理,还认为此处存在空值风险,所以这里报错
val upperCase = content.toUpperCase()
println(upperCase)
}
下面可通过!!操作符,告诉kt此处绝不为空,若出现风险则运行时抛出异常由自己承担;可以看出这样写法存在一定风险,因此最好提醒自己能否有更好实现,ef:
var content: String? = "hello"
fun main() {
if (content != null) {
printUpperCase()
}
}
fun printUpperCase() {
//此处!!操作符强调绝不为空,避免编译报错
val upperCase = content!!.toUpperCase()
println(upperCase)
}