Kotlin中使用Java数据类时引发的一个Bug

文章目录

基础复习:Kotlin语言中的对象比较

  • 比较对象的内容是否相等 (== 或者 equals )Kotlin 中的操作符 == 和 equals效果相同 ,都用于比较对象的内容是否相等, Kotlin中建议直接使用 ==
  • 比较对象的引用是否相等 ( === )Kotlin 中的操作符 === 用于比较对象的引用是否指向同一个地址,运行时如果是基本数据类型 === 等价于 ==

背景

如图效果,通过RecyclerView实现,对每个Item中前后数据对比来确定执行什么局部操作(如Item的insert、update、delete等),这里使用RecyclerView库中的DiffUtil.Callback()来进行的前后数据对比,如下:

kotlin 复制代码
class DataDiffUtil(private val oldModels: List<Any>, private val newModels: List<Any>) :DiffUtil.Callback() {

    /**
     * 旧数据
     */
    override fun getOldListSize(): Int = oldModels.size

    /**
     * 新数据
     */
    override fun getNewListSize(): Int = newModels.size

    /**
     * DiffUtil调用来决定两个对象是否代表相同的Item。true表示两个Item相同(表示View可以复用),false表示不相同(View不可以复用)
     * 例如,如果你的项目有唯一的id,这个方法应该检查它们的id是否相等。
     */
    override fun areItemsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
        return oldModels[oldItemPosition]::class.java == newModels[newItemPosition]::class.java
    }

    /**
     * 比较两个Item是否有相同的内容(用于判断Item的内容是否发生了改变),
     * 该方法只有当areItemsTheSame (int, int)返回true时才会被调用。
     */
    override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
        return oldModels[oldItemPosition] == newModels[newItemPosition]
    }

    /**
     * 该方法执行时机:areItemsTheSame(int, int)返回true 并且 areContentsTheSame(int, int)返回false
     * 该方法返回Item中的变化数据,用于只更新Item中变化数据对应的UI
     */
    override fun getChangePayload(oldItemPosition: Int, newItemPosition: Int): Any? {
        return super.getChangePayload(oldItemPosition, newItemPosition)
    }
}

以顶部的Item1模块举例,当服务端有新数据来时,通过下面方式进行更新:

kotlin 复制代码
/**
 * use[DiffUtil] 增量更新数据
 * @param newList 新数据
 */
fun submitList(newList: MutableList<Any>) {
    //传入新旧数据进行比对
    val diffUtil = DataDiffUtil(mModels, newList)
    //经过比对得到差异结果
    val diffResult = DiffUtil.calculateDiff(diffUtil)
    //NOTE:注意这里要重新设置Adapter中的数据
    setModels(newList)
    //将数据传给adapter,最终通过adapter.notifyItemXXX更新数据
    diffResult.dispatchUpdatesTo(this)
}

如果Item1前后数据是一样的,那么areContentsTheSame() 中的oldModels[oldItemPosition] == newModels[newItemPosition] 理论上返回的就是true了。那么Item1模块也就不会再刷新了。

实际跑起来能按我们的预期走吗?

问题出现

上述逻辑写的差不多了,还差Model数据类没有写出来,因为项目中是Kotlin & Java混用的,而Model数据类正好是用Java语言编写的:

kotlin 复制代码
public class VP2Model implements Serializable {
    public int id;
    public String content;
}

看上去一切都是OK的,但是运行之后发现出问题了,即使前后数据完全一样,仍然会进行Item1的刷新,说明areContentsTheSame()里的数据对比返回的是false,通过断点发现确实返回了false。

到这里不知道大家有没有发现问题所在 ?开始我以为是数据变了,但是通过Log打点发现前后数据也是一样的,那么明明是一样的,为什么对比会是不同呢?仔细一想明白了,问题出在Java语言上,出在VP2Model类中没有重新equals()方法

kotlin 复制代码
@Override
public boolean equals(@Nullable Object obj) {
    return super.equals(obj);
}

Java Model 类默认的equals()方法是比较的对象内存地址,刷新前后生成的显然不是同一个对象,那么前后内存地址对比返回的肯定是false了,问题就出在了这里!

如果我们使用 Kotlin 语言编写 Model 类就不会有这个问题,因为 Kotlin 编译器自动帮我们重写了equals()/hashCode()方法,如:

kotlin 复制代码
data class VP2Model(
    val id: Int = 0,
    val content: String = "",
)

注意这里要用data class开头才行,上述代码转换成Java后:

可以看到 Kotlin 编写的 Model 类自动帮我们实现了其中的equals()/hashCode()方法。

解决方式

已经知道问题出现的原因,那么解决方式就很简单了,比如下面几种解决方式:

方式一

重写Java Model 类中的 equals()方法,对每个字段进行对比,都相同返回 true,否则返回 false。

有一种快捷生成方式,在Mac版的 AS 中,可以使用 command + N 的方式生成,如下:

生成结果:

方式二

在使用的地方用 Kotlin语言编写Model 类进行转换,注意:这里一定要用data class 开头的声明,因为 Kotlin 编译器会自动帮我们重写 equals()/hashCode() 方法。

相关推荐
以后不吃煲仔饭2 分钟前
Java基础夯实——2.7 线程上下文切换
java·开发语言
进阶的架构师2 分钟前
2024年Java面试题及答案整理(1000+面试题附答案解析)
java·开发语言
The_Ticker8 分钟前
CFD平台如何接入实时行情源
java·大数据·数据库·人工智能·算法·区块链·软件工程
大数据编程之光30 分钟前
Flink Standalone集群模式安装部署全攻略
java·大数据·开发语言·面试·flink
爪哇学长44 分钟前
双指针算法详解:原理、应用场景及代码示例
java·数据结构·算法
ExiFengs1 小时前
实际项目Java1.8流处理, Optional常见用法
java·开发语言·spring
paj1234567891 小时前
JDK1.8新增特性
java·开发语言
繁依Fanyi1 小时前
简易安卓句分器实现
java·服务器·开发语言·算法·eclipse
慧都小妮子1 小时前
Spire.PDF for .NET【页面设置】演示:打开 PDF 时自动显示书签或缩略图
java·pdf·.net
m51271 小时前
LinuxC语言
java·服务器·前端