欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Kotlin對(duì)象比較注意點(diǎn)示例詳解

 更新時(shí)間:2022年04月17日 16:41:19   作者:HelloBan  
你知道在kotlin中,如何來比較對(duì)象相等嗎?下面這篇文章主要給大家介紹了關(guān)于Kotlin對(duì)象比較注意點(diǎn)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下

背景

  • 現(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)文章

最新評(píng)論