Kotlin對(duì)象比較注意點(diǎn)示例詳解
背景
- 現(xiàn)有一個(gè)StateFlow及其監(jiān)聽
private val stateFlow = MutableStateFlow(kotlin.Pair<String, ArrayList<String>>("abc", ArrayList()))
GlobalScope.launch { stateFlow.collect { // do something } }
- 更新ArrayList并嘗試emit
GlobalScope.launch { stateFlow.value.second.add("test") stateFlow.emit(stateFlow.value) }
實(shí)際上,collect并不會(huì)被調(diào)用
原因
MutableStateFlow真正的實(shí)現(xiàn)者是StateFlowImpl, emit方法代碼如下:
override suspend fun emit(value: T) { this.value = value }
查看value的set方法:
public override var value: T get() = NULL.unbox(_state.value) set(value) { updateState(null, value ?: NULL) }
private fun updateState(expectedState: Any?, newState: Any): Boolean { var curSequence = 0 var curSlots: Array<StateFlowSlot?>? = this.slots // benign race, we will not use it synchronized(this) { val oldState = _state.value if (expectedState != null && oldState != expectedState) return false // CAS support if (oldState == newState) return true // Don't do anything if value is not changing, but CAS -> true _state.value = newState curSequence = sequence ... 省略部分代碼 } }
其中"if (oldState == newState) return true"因emit前后是同一個(gè)對(duì)象,導(dǎo)致條件為true,那么,如果emit前后不是同一個(gè)對(duì)象,即可解決這個(gè)問題?
另一個(gè)問題
emit時(shí)嘗試以下代碼:
GlobalScope.launch { stateFlow.value.apply { stateFlow.emit(kotlin.Pair(first, second)) } }
實(shí)際上,上述代碼仍舊不能解決問題,因?yàn)閗otlin.Pair默認(rèn)重寫了equals方法,查看kotlin.Pair decompiled的Java文件
public final class Pair { public int hashCode() { Object var10000 = this.first; int var1 = (var10000 != null ? var10000.hashCode() : 0) * 31; Object var10001 = this.second; return var1 + (var10001 != null ? var10001.hashCode() : 0); } public boolean equals(@Nullable Object var1) { if (this != var1) { if (var1 instanceof Te.Pair) { Te.Pair var2 = (Te.Pair) var1; if (Intrinsics.areEqual(this.first, var2.first) && Intrinsics.areEqual(this.second, var2.second)) { return true; } } return false; } else { return true; } } }
其中Intrinsics.areEqual代碼如下:
public static boolean areEqual(Object first, Object second) { return first == null ? second == null : first.equals(second); }
故即使pair對(duì)象本身不一樣,但由于kotlin默認(rèn)重寫了equals方法,而pair.first與pair.second是一樣的,從而條件"if (oldState == newState) return true"成立
解決辦法
由于StateFlow源碼無法修改且是特定場(chǎng)景需求,故無法將判斷條件改為kotlin的"===";故使用android.util.Pair或者自定義java Pair class即可
結(jié)論
kotlin class默認(rèn)實(shí)現(xiàn)了equals方法,判斷的是內(nèi)容相等,而Java的class判斷的是地址相等,故判斷對(duì)象相等時(shí)需注意此細(xì)節(jié),根據(jù)需求來判斷地址相等(===)還是內(nèi)容相等(==)
到此這篇關(guān)于Kotlin對(duì)象比較注意點(diǎn)的文章就介紹到這了,更多相關(guān)Kotlin對(duì)象比較注意點(diǎn)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android listview數(shù)據(jù)顯示及提示信息的實(shí)例
這篇文章主要介紹了Android listview數(shù)據(jù)顯示及提示信息的實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-05-05Android開發(fā)之圖片旋轉(zhuǎn)功能實(shí)現(xiàn)方法【基于Matrix】
這篇文章主要介紹了Android開發(fā)之圖片旋轉(zhuǎn)功能實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Android基于matrix矩陣操作圖形變換的相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-09-09Android數(shù)據(jù)存儲(chǔ)方式操作模式解析
這篇文章主要為大家介紹了Android數(shù)據(jù)存儲(chǔ)方式操作模式解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08android實(shí)現(xiàn)簡(jiǎn)單拼圖游戲
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)簡(jiǎn)單拼圖游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03React?Native之在Android上添加陰影的實(shí)現(xiàn)
這篇文章主要介紹了React?Native之在Android上添加陰影的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03Android開發(fā)自學(xué)筆記(一):Hello,world!
這篇文章主要介紹了Android開發(fā)自學(xué)筆記(一):Hello,world!本文講解了創(chuàng)建HelloWorld工程、編寫代碼、啟動(dòng)模擬器等步驟,需要的朋友可以參考下2015-04-04Kotlin使用協(xié)程實(shí)現(xiàn)高效并發(fā)程序流程詳解
這篇文章主要介紹了Kotlin使用協(xié)程實(shí)現(xiàn)高效并發(fā)程序流程,協(xié)程屬于Kotlin中非常有特色的一項(xiàng)技術(shù),因?yàn)榇蟛糠志幊陶Z(yǔ)言中是沒有協(xié)程這個(gè)概念的。那么什么是協(xié)程呢?它其實(shí)和線程有點(diǎn)相似,可以簡(jiǎn)單地將它理解成一種輕量級(jí)的線程2023-01-01Android學(xué)習(xí)教程之懸浮窗菜單制作(9)
這篇文章主要為大家詳細(xì)介紹了Android學(xué)習(xí)教程之懸浮窗菜單制作方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11