Java 重写 equals 注意事项

自反性

满足条件: a.equals(a) = true 保证成立;

对称性

满足条件:如果 a.equals(b) = true 那么 b.equals(a)必须成立

违反规则案例

/*自定义类**/
public class CaseInsensitiveString {
    private final String s;
    public CaseInsensitiveString(String s) {
        this.s = Objects.requireNonNull(s);
    }
    @Override
    public boolean equals(Object obj) {
        if (obj instanceof CaseInsensitiveString) {
            return s.equalsIgnoreCase(((CaseInsensitiveString) obj).s);
        }
        if (obj instanceof String) {
            return s.equalsIgnoreCase((String) obj);
        }
        return false;
    }
}

/*main函数**/
public class equalsTest {
    public static void main(String[] args) {
        CaseInsensitiveString a = new CaseInsensitiveString("luoyonghui");
        String b = "luoyonghui";
        System.out.println(a.equals(b));
        System.out.println(b.equals(a));

        List list = new ArrayList<>();
        list.add(a);
        System.out.println(list.contains(b));
    }
}

输出日志结果

true
false
false

对输出结果解释如下:
a.equals(b) 比较时,使用 CaseInsensitiveString逻辑相等的规则,通过String#equalsIgnoreCase判断结果相同。但是在b.equals(a)比较时,使用String#equals()进行逻辑相等的判断。

/*String#equals实现代码如下**/
public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

如果不是String类型,直接返回false,所有导致上面的输出解决。同理,ArrayList#contains最终也是通过Object#equals进行逻辑判断,所以输出结果也为false

正确的实现方案

思路: 如果需要比对的两个对象的类型都不同,直接返回false。

  @Override
    public boolean equals(Object obj) {
        if (obj instanceof CaseInsensitiveString) {
            if (((CaseInsensitiveString) obj).s.equalsIgnoreCase(s)) {
                return true;
            }
        }
        return false;
    }

输出日志结果

false
false
false

你可能感兴趣的:(Java 重写 equals 注意事项)