背景
- private val stateFlow = MutableStateFlow(kotlin.Pair<String, ArrayList<String>>("abc", ArrayList()))
- GlobalScope.launch {
- stateFlow.collect {
- // do something
- }
- }
- 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对象比较注意点内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!