先以一段代码开始这篇blog。

01 public class Name {
02  
03   private String first; //first name
04   private String last;  //last name
05  
06   public String getFirst() {
07     return first;
08   }
09  
10   public void setFirst(String first) {
11     this.first = first;
12   }
13  
14   public String getLast() {
15     return last;
16   }
17  
18   public void setLast(String last) {
19     this.last = last;
20   }
21  
22   public Name(String first, String last) {
23     this.first = first;
24     this.last = last;
25   }
26  
27   @Override
28   public boolean equals(Object object) {
29     Name name = (Name) object;
30  
31     return first.equals(name.getFirst()) && last.equals(name.getLast());
32   }
33  
34   public static void main(String[] args) {
35     Map<Name, String> map = new HashMap<Name, String>();
36     map.put(new Name("mali", "sb"), "yes");
37      
38     System.out.println("is the key existed? ture or false? -> "
39         + map.containsKey(new Name("mali", "sb")));
40   }
41  
42 }

那输出结果是什么呢?类似这样的题目总能遇到,以前不知道有什么好考的,弱智?自己动手尝试了一次,发现结果不是自己想象的那样。本篇就用来揭示HashMap的equals与hashCode中你不知道的秘密。结果如下:

is the key existed? ture or false? -> false

对的,结果就是false,我很不理解为什么这样,已经重写了equals函数了啊!当时真心不服气,就在equals函数里面打了断点,然后更让我难以置信的事情发生了,断点处没有停。非常困惑,不过还好,jdk的源码在手上,去查了HashMap中containsKey函数的源码。源码结构如下图:

从图中可以看出,真正干活的是getEntry(Object key),重点看如下两行:

1 if (e.hash == hash &&
2                 ((k = e.key) == key || (key != null && key.equals(k))))
3     return e;

从if条件上看,是一个短路与,首先要判断两个对象的hash值是否相等。如果相等才进行后续的判断。或者换一个说法,在HashMap中只有两个对象的hash值相等的前提下才会执行equals方法的逻辑。关于这一点,有两个佐证。

在stackoverflow上找到一篇关于HashMap不执行equals方法的文章,回答中有明确给出这样的答案。Java HashMap.containsKey() doesn’t call equals()

自己编程验证。

在文章开头的基础上,做了点儿改进,输出两个对象的hash值,并且在equals方法中打印一行文字。如下:

01 public class Name {
02  
03   private String first; //first name
04   private String last;  //last name
05  
06   public String getFirst() {
07     return first;
08   }
09  
10   public void setFirst(String first) {
11     this.first = first;
12   }
13  
14   public String getLast() {
15     return last;
16   }
17  
18   public void setLast(String last) {
19     this.last = last;
20   }
21  
22   public Name(String first, String last) {
23     this.first = first;
24     this.last = last;
25   }
26  
27   @Override
28   public boolean equals(Object object) {
29     System.out.println("equals is running...");
30     Name name = (Name) object;
31  
32     return first.equals(name.getFirst()) && last.equals(name.getLast());
33   }
34  
35   public static void main(String[] args) {
36     Map<Name, String> map = new HashMap<Name, String>();
37     Name n1 = new Name("mali", "sb");
38     System.out.println("the hashCode of n1 : " + n1.hashCode());
39     map.put(n1, "yes");
40     Name n2 = new Name("mali", "sb");
41     System.out.println("the hashCode of n2 : " + n2.hashCode());
42     System.out.println("is the key existed? ture or false? -> "
43         + map.containsKey(n2));
44   }
45  
46 }

结果:

the hashCode of n1 : 1690552137

the hashCode of n2 : 1901116749

is the key existed? ture or false? -> false

从执行结果可以看出1、两个对象的hash值是不相同的;2、equals方法确实也没有执行。

再次对代码进行改进,加入重写的hashCode方法,如下,看看这次的结果会是怎样。

01 public class Name {
02  
03   private String first; //first name
04   private String last;  //last name
05  
06   public String getFirst() {
07     return first;
08   }
09  
10   public void setFirst(String first) {
11     this.first = first;
12   }
13  
14   public String getLast() {
15     return last;
16   }
17  
18   public void setLast(String last) {
19     this.last = last;
20   }
21  
22   public Name(String first, String last) {
23     this.first = first;
24     this.last = last;
25   }
26  
27   @Override
28   public boolean equals(Object object) {
29     System.out.println("equals is running...");
30     Name name = (Name) object;
31  
32     return first.equals(name.getFirst()) && last.equals(name.getLast());
33   }
34  
35   public int hashCode() {
36     System.out.println("hashCode is running...");
37     return first.hashCode() + last.hashCode();
38   }
39  
40   public static void main(String[] args) {
41     Map<Name, String> map = new HashMap<Name, String>();
42     Name n1 = new Name("mali", "sb");
43     System.out.println("the hashCode of n1 : " + n1.hashCode());
44     map.put(n1, "yes");
45     Name n2 = new Name("mali", "sb");
46     System.out.println("the hashCode of n2 : " + n2.hashCode());
47     System.out.println("is the key existed? ture or false? -> "
48         + map.containsKey(n2));
49   }
50  
51 }

结果:

hashCode is running...

the hashCode of n1 : 3347552

hashCode is running...

hashCode is running...

the hashCode of n2 : 3347552

hashCode is running...

equals is running...

is the key existed? ture or false? -> true

同样从结果中可以看出:在hash值相等的情况下,equals方法也执行了,HashMap的containsKey方法也像预想的那样起作用了。

结论:

在使用HashSet(contains也是调用HashMap中的方法)、HashMap等集合时,如果用到contains系列方法时,记得需同时重写equals与hashCode方法。

HashMap之equals和hashCode小陷阱的更多相关文章

  1. Java中equals,hashcode

         在Java语言中,Object对象中包含一个equals和hashCode方法,其中hashCode方法是由JVM本地代码(native code)实现的,返回值是一个有符号的32位整数,对 ...

  2. [转载] HashMap的工作原理-hashcode和equals的区别

    目录 前言 为什么需要使用Hashcode,可以从Java集合的常用需求来描述: 更深入的介绍 先来些简单的问题 HashMap的0.75负载因子 总结 我在网上看到的这篇文章,介绍的很不错,但是我看 ...

  3. == 和 equals,equals 与 hashcode,HashSet 和 HashMap,HashMap 和 Hashtable

    一:== 和 equals == 比较引用的地址equals 比较引用的内容 (Object 类本身除外) String obj1 = new String("xyz"); Str ...

  4. 堆、栈、内存分配、==、equals、hashcode详解(转载)

    问题的引入: 问题一:String str1 = "abc";String str2 = "abc";System.out.println(str1==str2 ...

  5. 如何在Java中避免equals方法的隐藏陷阱

    摘要 本文描述重载equals方法的技术,这种技术即使是具现类的子类增加了字段也能保证equal语义的正确性. 在<Effective Java>的第8项中,Josh Bloch描述了当继 ...

  6. Java == ,equals 和 hashcode 的区别和联系(阿里面试)

    今天阿里的人问我 equals 与hashcode的区别,我答不上来, 仔细查了一下,做了总结: (1) == 这是Java 比较内存地址,就是内存中的对象: java中的==是比较两个对象在JVM中 ...

  7. Object之equals和hashCode

    译者注 :你可能会觉得Java很简单,Object的equals实现也会非常简单,但是事实并不是你想象的这样,耐心的读完本文,你会发现你对Java了解的是如此的少.如果这篇文章是一份Java程序员的入 ...

  8. 关于重写equals()和hashCode()的思考

    最近这几天一直对equals()和hashCode()的事搞不清楚,云里雾里的. 为什么重写equals(),我知道. 但是为什么要两个都要重写呢,我就有点迷糊了,所以趁现在思考清楚后记录一下. 起因 ...

  9. 如何编写出高质量的 equals 和 hashcode 方法?

    什么是 equals 和 hashcode 方法? 这要从 Object 类开始说起,我们知道 Object 类是 Java 的超类,每个类都直接或者间接的继承了 Object 类,在 Object ...

随机推荐

  1. Sicily 8843 Ranking and Friendship

    http://soj.me/8843 题意:几个人想做好朋友,朋友之间相差位置小于等于k,且长度相同分析:排序,将长度相同的放在一起.若长度相同,第i个人能放进去的条件是位置相差下雨等于k.      ...

  2. Linux档案与文件系统的压缩与打包

    总结: 压缩指令为透过一些运算方法去将原本的档案进行压缩,以减少档案所占用的磁盘容量.压缩前与压缩后的档案所占用的磁盘容量比值,就可以被称为是“压缩比” 压缩的好处是可以减少磁盘容量的浪费,在www网 ...

  3. loj2062 [HAOI2016]地图

    ref #include <algorithm> #include <iostream> #include <cstdio> #include <cmath& ...

  4. Asp.net自定义控件开发任我行(6)-嵌入资源下

    摘要 上一章,我们讲了嵌入.css文件,这一章,我们来讲一下嵌入.js文件,也顺带一个嵌入Image文件 内容 我们前面的几章,一运行,下拉框就显示出来了,但是DropDwonList的下拉框是被隐藏 ...

  5. leetcode 【 Unique Paths 】python 实现

    题目: A robot is located at the top-left corner of a m x n grid (marked 'Start' in the diagram below). ...

  6. Effictive C++ 学习记录

    这是前段时间看的书,整理到这里吧,以后查看也方便. 这些条款需要反复查看. 条款01:视C++为一个语言联邦 条款02:尽量用const.enum.inline替换#define 条款03:尽可能的使 ...

  7. 使用 Region,RegionManager 在 XNA 中创建特殊区域(十八)

    平方已经开发了一些 Windows Phone 上的一些游戏,算不上什么技术大牛.在这里分享一下经验,仅为了和各位朋友交流经验.平方会逐步将自己编写的类上传到托管项目中,没有什么好名字,就叫 WPXN ...

  8. Python-S9——Day110-Git继续

    1 当日内容概要 2 内容回顾 3 Git版本控制之多人协同开发 4 Git版本控制之fork 5 版本控制之其他 6 Redis之字典基本操作 7 Django中操作Redis 8 Django缓存 ...

  9. 【LeetCode】Implement strStr()(实现strStr())

    这道题是LeetCode里的第28道题. 题目描述: 实现 strStr() 函数. 给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle ...

  10. cookie和session机制区别

    会话(Session)跟踪是Web程序中常用的技术,用来跟踪用户的整个会话.常用的会话跟踪技术是Cookie与Session.Cookie通过在客户端记录信息确定用户身份,Session通过在服务器端 ...