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<CaseInsensitiveString> 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
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,785评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,026评论 19 139
  • Overriding the equals method seems simple, but there are ...
    MrDcheng阅读 730评论 0 0
  • 有人说,穷人家的孩子早当家,有人说,女儿要富养。 你是怎样理解这个问题的? 下面我来说一下今天要讲的,男子要穷养,...
    离酒歌阅读 613评论 0 0
  • #三件事️️️️ 1.小打卡15mins❎ 2.阅读 30mins❎ 3.得到听一篇10mins✅ #其他自由时间...
    NancyLuo阅读 113评论 0 0