1. ? 和 !!
?
加在變量名后酸些,系統(tǒng)在任何情況不會(huì)報(bào)它的空指針異常。
!!
加在變量名后魄懂,如果對(duì)象為null,那么系統(tǒng)一定會(huì)報(bào)異常市栗!
上述是兩個(gè)符號(hào)的簡(jiǎn)單概念,為了更好的解釋這兩個(gè)概念填帽,我們先從java代碼入手,如下例:
ArrayList<String> myList = null; // 創(chuàng)建一個(gè)null的隊(duì)列
Log.d("TAG", "-->> List Size = " + myList.size());
這個(gè)例子中篡腌,執(zhí)行到 Log 打印隊(duì)列長(zhǎng)度時(shí)勾效,報(bào)錯(cuò)
“ Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'int java.util.ArrayList.size()' on a null object reference”
然而在 KT 中,在調(diào)用 myList
的時(shí)候在它后面加上一個(gè)問(wèn)號(hào) myList?.size()
层宫,則會(huì)在 myList
為null 的時(shí)候直接打印出null ,不會(huì)拋出 NullPointerException
萌腿。
在Android Studio 中,將上述代碼自動(dòng)轉(zhuǎn)換為 KT代碼寫(xiě)法后毁菱,如下:
val myList: ArrayList<String>? = null // 創(chuàng)建一個(gè)null的隊(duì)列
Log.d("TAG", "-->> List Size = " + myList!!.size)
編譯器為什么自動(dòng)將 myList.size()
變成了 myList!!.size
锌历,而不是加上 ?
呢峦筒?
這是因?yàn)榫幾g器在轉(zhuǎn)換時(shí)為了保證代碼轉(zhuǎn)換前后的一致性所造成的。換言之勘天,在Java 上出異常的捉邢,轉(zhuǎn)換到 KT 上,編譯器依然會(huì)讓它保持拋出異常伏伐,NullPointerException
也是如此。
綜上所述藐翎,使用 ?
時(shí),程序會(huì)進(jìn)行非空判斷吝镣,如為空,則返回 null 末贾,不會(huì)造成程序報(bào)錯(cuò)。但注意返回的是 null 而非 0 拱撵,如果用于判斷,參考 2中 ?:
的講解拴测;
使用 !!
時(shí),會(huì)對(duì)對(duì)象進(jìn)行非空判斷集索,并且會(huì)像 java 代碼一樣拋出異常。
2. ?:
對(duì)象A ?:
對(duì)象B 表達(dá)式抄谐,意思為,當(dāng)對(duì)象 A值為 null 時(shí)蛹含,那么它就會(huì)返回后面的對(duì)象 B。
val roomList: ArrayList<KTBean>? = null
if (roomList?.size > 0) {
Log.d("TAG", "-->> 列表數(shù)不是0")
}
如上段代碼浦箱,執(zhí)行時(shí)會(huì)發(fā)現(xiàn)如下提示
此時(shí)祠锣,就可以使用上述講到的 ?:
咽安,當(dāng) ?:
前面的對(duì)象為空時(shí),返回后面的值妆棒,如下:
val roomList: ArrayList<Room>? = null
val mySize= roomList?.size ?: 0
此時(shí),mySize
的值就為0糕珊,因?yàn)?roomList?.size
為空,可用于列表是否為空的判斷红选,如下:
val roomList: ArrayList<KTBean>? = null
if (roomList?.size ?: 0 > 0) { // 這一行添加了?:
Log.d("TAG", "-->> 列表數(shù)不是0")
}
至此,用上面的 喇肋?和 ?: 就可以避免程序中出現(xiàn)的 NullPointerException 。
3. ::
Kotlin 中 雙冒號(hào)操作符 表示把一個(gè)方法當(dāng)做一個(gè)參數(shù)蝶防,傳遞到另一個(gè)方法中進(jìn)行使用,通俗的來(lái)講就是引用一個(gè)方法慧脱。
詳情參考:https://blog.csdn.net/lv_fq/article/details/72869124
4. ->
fun <T, R> Collection<T>.fold(
initial: R,
combine: (acc: R, nextElement: T) -> R
): R {
var accumulator: R = initial
for (element: T in this) {
accumulator = combine(accumulator, element)
}
return accumulator
}
在上述代碼中,參數(shù) combine
具有函數(shù)類(lèi)型 (R, T) -> R
宗兼,因此 fold
接受一個(gè)函數(shù)作為參數(shù), 該函數(shù)接受類(lèi)型分別為 R
與 T
的兩個(gè)參數(shù)并返回一個(gè) R
類(lèi)型的值殷绍。 在 for 循環(huán)內(nèi)部調(diào)用該函數(shù),然后將其返回值賦值給 accumulator
主到。
5. == 和 ===
code 1
fun main(args: Array<String>) {
val a : Int = 1000
println(a == a) //true
println(a === a) //true
val a1 : Int = a
val a2 : Int = a
println(a1 == a2) //true
println(a1 === a2) //true
}
code 2
fun main(args: Array<String>) {
val a : Int = 1000
println(a == a) //true
println(a === a) //true
val a1 : Int? = a
val a2 : Int? = a
println(a1 == a2) //true
println(a1 === a2) //false
}
在Kotlin中,===
表示比較對(duì)象地址登钥,==
表示比較兩個(gè)值大小。
所以無(wú)論是 a == a
還是 a === a
都是返回true牧牢,因?yàn)槭峭粋€(gè)變量,數(shù)值大小和地址都是相等的塔鳍。
現(xiàn)在重點(diǎn)看 a1 和 a2,這里的把 a 分別賦給 a1 和 a2轮纫。
code 1 和 code 2 的不同點(diǎn)在于 a1 和 a2 的類(lèi)型。一個(gè)是Int
掌唾,一個(gè)是Int?
。它們的區(qū)別如下:
如果我們使用的是
val a : Int = 999
這種方式糯彬,這時(shí)的a其實(shí)就是個(gè)數(shù)值, 不涉及裝箱的問(wèn)題, 也就不是對(duì)象。
而如果我們使用的是val a: Int? = 999
這種方式,這時(shí)的a是一個(gè)Int型對(duì)象, 因?yàn)樗婕暗窖b箱問(wèn)題情连。
code 1 中 a1 和 a2 都沒(méi)有裝箱览效,所以不是對(duì)象,只是數(shù)值锤灿,所以數(shù)值大小和地址都是相等的。而 code 2 中 a1 和 a2 涉及到裝箱但校,已經(jīng)變成了對(duì)象,此時(shí)它們的數(shù)值仍然相等状囱,但地址已經(jīng)不同了(因?yàn)槭遣煌瑢?duì)象)。
code 3
fun main(args: Array<String>) {
val a : Int? = 1000
println(a == a) //true
println(a === a) //true
val a1 : Int? = a
val a2 : Int? = a
println(a1 == a2) //true
println(a1 === a2) //true
}
code 3 和 code 2 做比較亭枷,發(fā)現(xiàn)將a也裝箱后,a1 === a2
返回 true叨粘,這是為什么呢?
因?yàn)檫@里的a經(jīng)過(guò)裝箱后本身已經(jīng)一個(gè)對(duì)象升敲,所以賦給a1和a2的時(shí)候是把直接把對(duì)象a賦給它們,所以此時(shí)a1和a2指的是同一個(gè)對(duì)象(對(duì)象a)驴党。既然是同一個(gè)對(duì)象,那么數(shù)值大小和地址肯定都是相等的了(也就是說(shuō)a,a1和a2這三個(gè)對(duì)象指向同一處地址设江,所以其實(shí) a === a1
和a === a2
也是返回true)。
code 4
fun main(args: Array<String>) {
val a : Int = 100
println(a == a) //true
println(a === a) //true
val a1 : Int? = a
val a2 : Int? = a
println(a1 == a2) //true
println(a1 === a2) //true
}
code 4 和 code 2 做比較叉存,明明只是改了一下a的值,為什么就會(huì)產(chǎn)生不同的結(jié)果呢歼捏?
這里跟 Java 中是一樣的,在范圍是 [-128, 127] 之間的數(shù)裝箱時(shí)并不會(huì)創(chuàng)建新的對(duì)象瞳秽,所以這里a1和a2裝箱后的對(duì)象是同一個(gè),a1 === a2
也就返回true了练俐。這里改為128或-129就又會(huì)變成false了。