一尘不染

在java中如何比较两个对象是否相等?

java

在java中如何比较两个对象是否相等?何时使用equals和hashcode?


阅读 1844

收藏
2020-01-07

共1个答案

一尘不染

理论(针对语言律师和数学倾向者):

equals()javadoc)必须定义一个等价关系(它必须是自反的,对称的和可传递的)。另外,它必须是一致的(如果未修改对象,则它必须保持返回相同的值)。此外,o.equals(null)必须始终返回false。

hashCode()javadoc)也必须是一致的(如果未根据修改对象equals(),则它必须保持返回相同的值)。

该关系的两种方法之间是:

每当a.equals(b),则a.hashCode()必须与相同b.hashCode()。

在实践中:

如果覆盖一个,则应覆盖另一个。

使用用于计算的相同字段集equals()进行计算hashCode()。

使用优秀的辅助类EqualsBuilderHashCodeBuilder,Apache Commons Lang 。一个例子:

public class Person {
    private String name;
    private int age;
    // ...

    @Override
    public int hashCode() {
        return new HashCodeBuilder(17, 31). // two randomly chosen prime numbers
            // if deriving: appendSuper(super.hashCode()).
            append(name).
            append(age).
            toHashCode();
    }

    @Override
    public boolean equals(Object obj) {
       if (!(obj instanceof Person))
            return false;
        if (obj == this)
            return true;

        Person rhs = (Person) obj;
        return new EqualsBuilder().
            // if deriving: appendSuper(super.equals(obj)).
            append(name, rhs.name).
            append(age, rhs.age).
            isEquals();
    }
}

还请记住:

使用基于哈希的CollectionMap(例如HashSetLinkedHashSetHashMapHashtableWeakHashMap)时,请确保放入对象的关键对象的hashCode()永远不会在对象位于集合中时改变。确保这一点的防弹方法是使您的钥匙不可变,这还有其他好处。

2020-01-08