从String的equals方法说起

缘起
调试一段代码判断字符串是否相同时,咋一看去字符串分明是一样的,怎么就是不相等呢?debug一开,瞬间明朗,原来配置文件中多了一个空格。

  1. 先看下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;
    }
  1. 讲讲下这段代码
    • ==判断是否是同一个对象,即比较引用地址
    • 判断对象类型是否为目标对象
    • 比较字符串长度是否相同
    • 注意比较各个字符是否相同

细看来这段断码写的也是颇有意味,先比较地址,再判断类型,最后比较各个字符,一气呵成,不带半分冗余。

  1. 再看一下Java Bean自动生成的equals方法

public class RequestInfo implements Serializable{
    private String ip;

    public String getIp() {
        return ip;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;

        RequestInfo that = (RequestInfo) o;

        return !(ip != null ? !ip.equals(that.ip) : that.ip != null);

    }

    @Override
    public int hashCode() {
        return ip != null ? ip.hashCode() : 0;
    }

    public void setIp(String ip) {
        this.ip = ip;
    }



}

  • 单独看一下这个equals方法
    • 与String的equals方法有异曲同工之妙
    • 先比较地址,然后是对象类类型,最后比较各属性值

public boolean equals(Object o) {
    if (this == o)
         return true;

    if (o == null || getClass() != o.getClass())
         return false;

    RequestInfo that = (RequestInfo) o;

    return !(ip != null ? !ip.equals(that.ip) : that.ip != null);

}

  1. 总结一下equals方法
    • 精髓在于从大到小注意缩小比较范围
    • 逐步确定两者是否相同
  2. 番外
    • 常言道覆写equals时一定要吧hashcode也一并覆写掉
    • 那这是为什么呢?
  3. 番外解惑
    • 既然是hashcode自然与集合关系密切,这也是其需一起覆写的原因所在
    • 正所谓源码之下无秘密,来看一段HashSet源码
private transient HashMap<E,Object> map;
...  ...
public boolean add(E e) {
    return map.put(e, PRESENT)==null;
}

public boolean remove(Object o) {
    return map.remove(o)==PRESENT;
}

  • 此时需要继续跟进到HashMap的源码
    • 看到static final int hash(Object key)这个方法,一切就得到了一个合理的解释
    • hash类集合对象在add/remove等操作时,需要根据hash值和equals方法共同确定待比较对象与集合对象是否相同
    • 想进一步了解HashMap的put过程可以参考源码深入研究
 public V put(K key, V value) {
        return putVal(hash(key), key, value, false, true);
    }

 static final int hash(Object key) {
        int h;
        return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
    }

PS:

  • 如果在计算hashcode时使用了某对象属性,如果修改该属性后,再去remove Hash类集合对象时会失败,因为修改属性值后,hashcode也就改变了,集合根据hashcode和equals方法判定此对象不存在

参考:
http://blog.csdn.net/afgasdg/article/details/6889383

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,767评论 18 399
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,372评论 11 349
  • Tip:笔者马上毕业了,准备开始 Java 的进阶学习计划。于是打算先从 String 类的源码分析入手,作为后面...
    石先阅读 12,056评论 16 58
  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 1,518评论 0 3
  • 这两天看到不少好文章,“匠人精神”、“精力”、“专心”、“跨学科”等词汇一直不断地在我眼前闪过。 下午在简书上看到...
    GuangHui阅读 481评论 0 2