自从Google于2017年正式宣布Kotlin是Android开发的首选语言以来,越来越多的开发者开始使用Kotlin。而数据类(data class)是Kotlin的一大特色。 他们省去了手动编写大量模板代码的工作,例如equals(),hash()和toString()的覆写都不是必须的了。数据类会自动提供正确的覆写。但是,在实际应用中,也会出现例外的情况。让我们看下面这段代码:
data class NumArray(val name: String, val values: IntArray)
对上述代码编译之后,会产生如下等价的Java代码:
public class NumArray {
private final String name;
private final int[] values;
...
@Override
public boolean equals(final Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
final NumArray numArray = (NumArray) o;
if (name != null ? !name.equals(numArray.name) : numArray.name != null) return false;
return Arrays.equals(values, numArray.values);
}
@Override
public int hashCode() {
int result = name != null ? name.hashCode() : 0;
result = 31 * result + Arrays.hashCode(values);
return result;
}
}
可以看到,Kotlin编译器为我们自动生成了equals和hashCode方法。接下来,我们通过下面的代码来测试NumArray类的实现。
val n1 = NumArray("1", intArrayOf(1,2,3,4))
val n2 = NumArray("1", intArrayOf(1,2,3,4))
val result = n1==n2
println("result=$result")
令人惊讶的是,上述程序打印出来的输出值居然是false。这是由于在Kotlin中,equals()总是用来比较数组。但是,Java里有一个bug,那就是在比较数组的时候,equals()只会比较引用,而不是比较数组的内容。由于我们创建了两个不同的数组实例,因此上述程序会输出false。
我们可以通过覆写equals()和hash()解决上述问题,但是就无法得到一个简洁的data class实现了。一个更为巧妙的解决方案是使用List来代替数组类型。因为,对于Collection类型及其子类来说,equals()的比较是基于内容而不是引用的。