如上图所未,一个空可的String变量,调用了两个函数,一个没报错,一个报错,后来发现hashCoe()函数没报错是因为它是一个可空对象的扩展函数,如下:
public inline fun Any?.hashCode(): Int = this?.hashCode() ?: 0
所以,为什么可以在一个可空的String上调用hasCode()函数的原因了。哎,Kotlin为什么要搞这种特殊啊,刚开始不了解情况我还以为是我的编译环境出问题了,这让人很疑惑啊!为什么要搞这种特殊,让人摸不着头脑!!
仔细想想,它是肯定为了某些好处才这样设计的,即不影响运行的情况下才会这样设计,比如str.hashCode(),如果str对象为null,则返回0,对运行结果是没有影响的,又比如str.toString(),如果str为null,则返回字符串"null",你可能会想,如果我把str和null对象进行比较的话那不就出问题了吗?null对象和字符串null可不是一个概念啊,是的没错,但是Kotlin会对equals也有相应的扩展函数,如下:
public actual fun String?.equals(other: String?, ignoreCase: Boolean = false): Boolean {
if (this === null)
return other === null
return if (!ignoreCase)
(this as java.lang.String).equals(other)
else
(this as java.lang.String).equalsIgnoreCase(other)
}
所以,我们就大胆的相信Kotlin吧,只要在IDE中没有报错,就不用担心出现空指针或其他的问题了!
这种对可空对象的扩展有时候用起来也很方便的,示例如下:
val list: List<String>? = null
if (list.isNullOrEmpty()) {
// TODO todo something
}
如果没有这个可空扩展函数的话,需要这样写代码:
val list: List<String>? = null
if (list?.isEmpty() == true) {
// TODO todo something
}
// 或者这样写:
if (list == null || list.isEmpty()) {
// TODO todo something
}
感觉起来是不是麻烦一点呢!
所以,我们平时在开发的时候,也可以学一下这招,即在获取某个对象的某个属性的时候,即使这个对象为空了,我们也需要使用一个默认值,有这种需求时就可以创建可空的扩展函数了,比如,我们需要获取一个人的id属性,如果这个人是null,则使用默认值-1作为id值,示例如下:
data class Person(val userId: Int)
fun Person?.getUserId(): Int = this?.userId ?: -1
fun main() {
val person: Person? = null
val userId = person.getUserId() // 等效于:val userId = person.userId ?: -1
println(userId) // 输出:-1
}