返回顶部
首页 > 资讯 > 移动开发 >Kotlin对象比较注意点示例详解
  • 767
分享到

Kotlin对象比较注意点示例详解

2024-04-02 19:04:59 767人浏览 独家记忆
摘要

目录背景原因另一个问题解决办法结论背景 现有一个StateFlow及其监听 private val stateFlow = MutableStateFlow(Kotlin.Pair&

背景

  • 现有一个StateFlow及其监听
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)
}

实际上,collect并不会被调用

原因

MutableStateFlow真正的实现者是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前后是同一个对象,导致条件为true,那么,如果emit前后不是同一个对象,即可解决这个问题?

另一个问题

emit时尝试以下代码:

GlobalScope.launch {
    stateFlow.value.apply {
        stateFlow.emit(kotlin.Pair(first, second))
    }
}

实际上,上述代码仍旧不能解决问题,因为kotlin.Pair默认重写了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对象本身不一样,但由于kotlin默认重写了equals方法,而pair.first与pair.second是一样的,从而条件"if (oldState == newState) return true"成立

解决办法

由于StateFlow源码无法修改且是特定场景需求,故无法将判断条件改为kotlin的"===";故使用Android.util.Pair或者自定义java Pair class即可

结论

kotlin class默认实现了equals方法,判断的是内容相等,而Java的class判断的是地址相等,故判断对象相等时需注意此细节,根据需求来判断地址相等(===)还是内容相等(==)

到此这篇关于Kotlin对象比较注意点的文章就介绍到这了,更多相关Kotlin对象比较注意点内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Kotlin对象比较注意点示例详解

本文链接: https://lsjlt.com/news/146257.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作