Kotlin中?和!!的區別詳細對比

前言

很多同學剛上手使用Kotlin知道它有針對Java NullPointerException的管理,而在Kotlin中?和!!均是和NullPointerException有關系,可他們的區別到底是什麼呢?為什麼別人開發的項目中出現瞭好多”?”,而我讀起來卻滿臉問號。

不懂就問百度呀,確實有人解釋它們的區別,比如:

這是輸入”kotlin ?和!!”搜索到的百度第一條答案,確實這位說的沒錯。不過我覺得對於一個剛接觸KT(Kotlin)的新手來說,他恐怕需要有漢語八級才能透徹理解這兩句話的意思。

先闡述兩個概念:

  • “?”加在變量名後,系統在任何情況不會報它的空指針異常。
  • “!!”加在變量名後,如果對象為null,那麼系統一定會報異常!

在Java上出異常的,轉化到KT上,編譯器任然會讓他保持拋出異常,NullPointerException也是如此。

所以結合上下文可以看得出,!!加上去後好像並沒有和之前Java代碼有什麼區別嘛,該null的地方任然會拋出異常。所以大多數情況下都會使用?來檢測null,輪不到!!出場。!!隻會在你需要對某對象進行非空判斷,並且需要拋出異常時才會使用到。

那我們接下來著重講解一下?到底怎麼用。

在聲明對象時,把它跟在類名後面,表示這個類允許為null;

在調用對象時,把它跟在對象後面,表示如果為null程序就會視而不見。

如下列代碼:

// 這是聲明一個變量,問號跟在類名後面
var room: Room? = Room()

private fun checkRoom() {
    // 因為加上瞭問號,所以可以任意的把room變成空
    room = null

    // 因為在調用時加上瞭問號,所以程序不會拋出異常
    Log.d("TAG", "-->> room name = ${room?.roomName}")
}

再舉個不用?的例子:
    // 這樣程序就默認的給room加上瞭!!,從此以後room不允許為null
var room: Room = Room()

private fun checkRoom() {
    // 當把null賦給room時,從編譯的時候就已經不通過
    room = null
    // 並且編譯器建議把對象後面的問號刪除,因為這個對象永遠不為空
    Log.d("TAG", "-->> room name = ${room.roomName}")
}

所以加上?是一種安全的寫法,它體現瞭Kotlin null safety的特性。

KT的語法很靈動,定義參數還可以寫成

val room: Room? = Room()    // 先實例化一個room,並且room可以為空
val room: Room? = null  // 不實例化瞭,開始room就是空的

val room: Room = Room()   // 實例化一個room,並且room永遠不能為空
val room = Room()   // 和上一行代碼一樣,是KT最常用的簡寫語法

然而加上問號以後程序就萬事大吉永遠擺脫瞭NullPointerException的煩惱?我們再看下一段代碼:

 `   val roomList: ArrayList<Room>? = null
    if (roomList?.size > 0) {
        Log.d("TAG", "-->> 房間數不是0")
    }
    `

當我們判斷list.size的時候,編譯器會告訴我們”Operator call corresponds to a dot-qualified call ‘roomList?.size.compareTo(0)’ which is not allowed on a nullable receiver ‘roomList?.size’.”。大概意思是,當roomList為null的時,它的size返回就是”null”,但是”null”不可以和int值比大小,所以編譯器建議我們寫成roomList?.size!! > 0。

沒錯,經過編譯器的建議加上瞭!!,我們程序運行到這行代碼,roomList為null時它一定會報異常。所以是不是必須得在外面套一層if(roomList != null)這種Java常見語句才能避免異常嗎?

當然Kotlin不會讓程序出現這種囉嗦的代碼,所以裡面提供瞭對象A ?: 對象B表達式,並且取消瞭Java中的條件表達式 ? 表達式1 : 表達式2這個三元表達式。

?:表示的意思是,當對象A值為null的時候,那麼它就會返回後面的對象B。

    `val roomList: ArrayList<Room>? = null
    val mySize= roomList?.size ?: 0  

總結

到此這篇關於Kotlin中?和!!區別的文章就介紹到這瞭,更多相關Kotlin中?和!!區別內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: