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編程中ViewPage判斷左右滑動(dòng)方向的方法
這篇文章主要介紹了Android編程中ViewPage判斷左右滑動(dòng)方向的方法,涉及Android中ViewPage針對(duì)滑動(dòng)判定的相關(guān)技巧,非常簡(jiǎn)單實(shí)用,需要的朋友可以參考下2015-10-10
Android的RV列表刷新詳解Payload與Diff方式異同
這篇文章主要為大家介紹了Android的RV列表刷新詳解Payload與Diff方式異同,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10
純android代碼實(shí)現(xiàn)九宮格手勢(shì)密碼
這篇文章主要為大家詳細(xì)介紹了純android代碼實(shí)現(xiàn)九宮格手勢(shì)密碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07
Android中SeekBar和RatingBar用法實(shí)例分析
這篇文章主要介紹了Android中SeekBar和RatingBar用法,結(jié)合實(shí)例形式分析了SeekBar和RatingBar的功能、定義與簡(jiǎn)單使用方法,需要的朋友可以參考下2016-06-06
Android鬧鐘機(jī)制實(shí)現(xiàn)定時(shí)任務(wù)功能
這篇文章主要為大家詳細(xì)介紹了Android鬧鐘機(jī)制實(shí)現(xiàn)定時(shí)任務(wù)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
Android 友盟第三方登錄與分享的實(shí)現(xiàn)代碼
這篇文章主要介紹了Android 友盟第三方登錄與分享的實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
Android直播app送禮物連擊動(dòng)畫效果(實(shí)例代碼)
最近在做公司的直播項(xiàng)目,需要實(shí)現(xiàn)一個(gè)觀看端連擊送禮物的控件,下面給大家分享實(shí)例代碼,需要的的朋友參考下吧2017-07-07
Android常用控件ImageSwitcher使用方法詳解
這篇文章主要為大家詳細(xì)介紹了Android常用控件ImageSwitcher的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08

