经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » Kotlin » 查看文章
Kotlin对象比较注意点示例详解
来源:jb51  时间:2022/4/18 10:11:02  对本文有异议

背景

  • 现有一个StateFlow及其监听
  1. private val stateFlow = MutableStateFlow(kotlin.Pair<String, ArrayList<String>>("abc", ArrayList()))
  1. GlobalScope.launch {
  2. stateFlow.collect {
  3. // do something
  4. }
  5. }
  • 更新ArrayList并尝试emit
  1. GlobalScope.launch {
  2. stateFlow.value.second.add("test")
  3. stateFlow.emit(stateFlow.value)
  4. }

实际上,collect并不会被调用

原因

MutableStateFlow真正的实现者是StateFlowImpl, emit方法代码如下:

  1. override suspend fun emit(value: T) {
  2. this.value = value
  3. }

查看value的set方法:

  1. public override var value: T
  2. get() = NULL.unbox(_state.value)
  3. set(value) { updateState(null, value ?: NULL) }
  1. private fun updateState(expectedState: Any?, newState: Any): Boolean {
  2. var curSequence = 0
  3. var curSlots: Array<StateFlowSlot?>? = this.slots // benign race, we will not use it
  4. synchronized(this) {
  5. val oldState = _state.value
  6. if (expectedState != null && oldState != expectedState) return false // CAS support
  7. if (oldState == newState) return true // Don't do anything if value is not changing, but CAS -> true
  8. _state.value = newState
  9. curSequence = sequence
  10. ... 省略部分代码
  11. }
  12. }

其中"if (oldState == newState) return true"因emit前后是同一个对象,导致条件为true,那么,如果emit前后不是同一个对象,即可解决这个问题?

另一个问题

emit时尝试以下代码:

  1. GlobalScope.launch {
  2. stateFlow.value.apply {
  3. stateFlow.emit(kotlin.Pair(first, second))
  4. }
  5. }

实际上,上述代码仍旧不能解决问题,因为kotlin.Pair默认重写了equals方法,查看kotlin.Pair decompiled的Java文件

  1. public final class Pair {
  2. public int hashCode() {
  3. Object var10000 = this.first;
  4. int var1 = (var10000 != null ? var10000.hashCode() : 0) * 31;
  5. Object var10001 = this.second;
  6. return var1 + (var10001 != null ? var10001.hashCode() : 0);
  7. }
  8.  
  9. public boolean equals(@Nullable Object var1) {
  10. if (this != var1) {
  11. if (var1 instanceof Te.Pair) {
  12. Te.Pair var2 = (Te.Pair) var1;
  13. if (Intrinsics.areEqual(this.first, var2.first) && Intrinsics.areEqual(this.second, var2.second)) {
  14. return true;
  15. }
  16. }
  17. return false;
  18. } else {
  19. return true;
  20. }
  21. }
  22. }

其中Intrinsics.areEqual代码如下:

  1. public static boolean areEqual(Object first, Object second) {
  2. return first == null ? second == null : first.equals(second);
  3. }

故即使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!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号