问题描述
示例:
fun main(){
var userInput: String?
//userInput = null
userInput = "asbdef"
var inputLength:Int? = userInput!!.length
println("Length of the string is :"+inputLength)
}
输出: 字符串的长度是:6
fun main(){
var userInput: String?
userInput = null
//userInput = "asbdef"
var inputLength:Int? = userInput!!.length
println("Length of the string is :"+inputLength)
}
输出: 未解析的参考:长度
我想知道为什么会出现编译错误?
如果我只是将(!!)运算符替换为(?),则编译效果很好,但将输出打印为null。
PS:我是科特林的新手
解决方法
如果左侧的求值结果为?.
,则null
运算符会短路。因此,nullVariable?.length
的结果为null
,并且从未评估.length
。您的第一个示例有效地做到了:
println("Length of the string is :" + null)
!!
运算符说:“如果左侧为空,则抛出NullPointerException
。否则,我们知道右侧不为空”。如果您稍微更改第二个示例,您会看到以下信息:
val userInput: String? = null
val inputLength:Int = userInput!!.length // throws NullPointerException
但是,我不确定您为什么得到Unresolved reference: length
。当您直接将null
分配给userInput
时,似乎编译器正在进行一些优化,因此与其将其编译为在运行时抛出NPE的String?
,不如说它知道在编译时仅null
(而不是String?
),因此不能拥有length
属性,因为null
不是引用。
您可以通过函数调用添加抽象层来强制其使您进入NullPointerException
:
fun main(){
var userInput: String? = "foo"
userInput = alwaysNull()
val inputLength:Int = userInput!!.length // NullPointerException
}
fun alwaysNull(): String? = null
在Null Safety文档中,我看不到任何关于一行初始化/通过函数调用与直接将null
分配给var之间的行为差异的信息,因此在获得Unresolved reference
的引擎盖只是一个猜测。