溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

詳解Kotlin的空指針處理

發(fā)布時(shí)間:2020-09-26 13:23:49 來(lái)源:腳本之家 閱讀:149 作者:還不走 欄目:編程語(yǔ)言

詳解Kotlin的空指針處理

Kotlin的空指針處理相比于java有著極大的提高,可以說(shuō)是不用擔(dān)心出現(xiàn)NullPointerException的錯(cuò)誤,kotlin對(duì)于對(duì)象為null的情況有嚴(yán)格的界定,編碼的階段就需要用代碼表明引用是否可以為null,為null的情況需要強(qiáng)制性的判斷處理。

咋看一下這些在java里面其實(shí)也有,問(wèn)題是一般開(kāi)發(fā)中不寫(xiě)也是可以的(大部分開(kāi)發(fā)不會(huì)花很多時(shí)間考慮這些),等出了空指針錯(cuò)誤再一個(gè)個(gè)打補(bǔ)丁。這樣往往會(huì)遺漏很多空指針,后期的解決僅僅是做一個(gè)if判斷,沒(méi)有從根源解決問(wèn)題

變量需要知道是否可以為null

第一個(gè)與java的不同,kotlin聲明引用時(shí)不可以直接賦值為null

var helloA: String=null //編譯器直接報(bào)錯(cuò) null can not be value of a non null type string

這里插一句,kotlin的成員變量(全局變量)必須要初始化甚至是基本數(shù)據(jù)類(lèi)型都要手動(dòng)給一個(gè)初始值,局部變量可以不用初始化,上面的例子是成員變量的聲明。

編譯器直接表示hello是一個(gè)non null type你不可以直接賦一個(gè)null值。對(duì)于我們java原住民來(lái)說(shuō)聲明變量時(shí)如果不去賦值,編譯器會(huì)默認(rèn)賦null(除去基本數(shù)據(jù)類(lèi)型),在kotlin這是不允許的。

類(lèi)型?

當(dāng)某個(gè)變量的值可以為 null 的時(shí)候,必須在聲明處的類(lèi)型后添加 ? 來(lái)標(biāo)識(shí)該引用可為空。

這是官方文檔的說(shuō)明,也就是說(shuō)上面的變量聲明只需要在String類(lèi)型后面加一個(gè)?就可以解決這個(gè)錯(cuò)誤。

var helloB : String? = null

如果不用?也可以直接初始化來(lái)解決問(wèn)題

var helloA : String = "hello"

String和String?

helloA 和 helloB2個(gè)變量的類(lèi)型分別為String和String?,這2種類(lèi)型的區(qū)別在于引用時(shí)是否空安全,String?代表可能為空,引用不安全,String代表不會(huì)為空,引用安全。比如調(diào)用一下helloB的方法

詳解Kotlin的空指針處理

可以看到有些方法是灰的代表不能直接調(diào)用,有些是白的代表可以直接調(diào)用,為什么可以直接調(diào)用后面再講

helloB.length //Only safe (?.) or non-null asserted (!!.) calls are allowed on a nullable receiver of type String?
這里的錯(cuò)誤解釋簡(jiǎn)單的理解就是helloB是一個(gè)可能為空的類(lèi)型(String?),對(duì)于可空類(lèi)型你必須做判空的處理才能調(diào)用方法。有沒(méi)有愛(ài)上kotlin? wtf太厲害了

如何做判空處理

最簡(jiǎn)單的方法就是使用?.來(lái)調(diào)用方法。這個(gè)符號(hào)的意義是,當(dāng)helloB不為空調(diào)用.length方法,返回Int值,如果helloB為空則返回null。

helloB?.length

注意這里的返回值類(lèi)型為Int? 依然是可以為空的類(lèi)型

不推薦的寫(xiě)法!!這個(gè)操作符可以說(shuō)是為了空指針錯(cuò)誤愛(ài)好者準(zhǔn)備的,使用了!!操作符代表你不關(guān)心變量的空判斷可以報(bào)出NullPointerException,又或者你有絕對(duì)的自信這里肯定不為空,可以放心的調(diào)用,可以說(shuō)!!放棄了判空的檢查

helloB!!.length

Elvis 操作符?:

回到?.的調(diào)用上來(lái),這個(gè)調(diào)用方式存在一個(gè)讓人不安的處理,就是在變量為null的情況下,會(huì)直接返回null,這樣空指針的隱患還在。

var l : Int = helloB?.length //報(bào)錯(cuò)類(lèi)型不匹配 Int? 和 Int

修正的話需要通過(guò)if判斷來(lái)進(jìn)行判空處理

var l: Int = if (helloB?.length != null) helloB.length else -1

這種寫(xiě)法可以簡(jiǎn)化成Elvis 操作符?:

var le: Int = helloB?.length ?: 0

當(dāng)?:左邊非空返回左邊的值,左邊為空則返回右邊的值

最后前文提到的當(dāng)helloB為空時(shí)有些方法仍然可以調(diào)用比如equals方法

helloB.equals(helloA)

即使helloB為可空類(lèi)型依然可以調(diào)用equals方法,但是這里的equals方法并不是String.java的equals方法,而是StringJvm.kt中的equals方法

public 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)
}

關(guān)鍵代碼

this===null
return other === null

這個(gè)方法對(duì)變量本身為null的情況已經(jīng)做了判斷,同時(shí)方法寫(xiě)成了String?.equals可以讓String?類(lèi)型的對(duì)象直接調(diào)用這個(gè)equals方法

感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI