平时使用map时都是用JAVA原生的类型,所以很少关注到hashcode()和equals()的方法的内部实现。近期实现一个小工具,涉及到自己写的类的查找比对,又再次重温了相关的知识。

上简单示例代码,比对自己覆写和不覆写的差异

不覆写hashcode()和equals():

 import java.util.HashMap;
import java.util.Map; public class HashCode_Equals { public int a;
public int b; public static void main(String[] args) { Map<HashCode_Equals, String> map = new HashMap<HashCode_Equals, String>(); HashCode_Equals obj1 = new HashCode_Equals();
obj1.a = 1;
obj1.b = 1;
map.put(obj1, "test"); HashCode_Equals obj2 = new HashCode_Equals();
obj2.a = 1;
obj2.b = 1; System.out.println("obj1.equals(obj2)=" + obj1.equals(obj2));
System.out.println("map.get(obj2)=" + map.get(obj2));
}
}

执行结果:

obj1.equals(obj2)=false
map.get(obj2)=null

仅覆写equals()方法

 import java.util.HashMap;
import java.util.Map; public class HashCode_Equals { public int a;
public int b; @Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
HashCode_Equals other = (HashCode_Equals) obj;
if (a != other.a)
return false;
if (b != other.b)
return false;
return true;
} public static void main(String[] args) { Map<HashCode_Equals, String> map = new HashMap<HashCode_Equals, String>(); HashCode_Equals obj1 = new HashCode_Equals();
obj1.a = 1;
obj1.b = 1;
map.put(obj1, "test"); HashCode_Equals obj2 = new HashCode_Equals();
obj2.a = 1;
obj2.b = 1; System.out.println("obj1.equals(obj2)=" + obj1.equals(obj2));
System.out.println("map.get(obj2)=" + map.get(obj2));
} }

执行结果:

obj1.equals(obj2)=true
map.get(obj2)=null

同时覆写hashcode()和equals()(借用eclipse集成的功能自动生成)

 import java.util.HashMap;
import java.util.Map; public class HashCode_Equals { public int a;
public int b; @Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + a;
result = prime * result + b;
return result;
} @Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
HashCode_Equals other = (HashCode_Equals) obj;
if (a != other.a)
return false;
if (b != other.b)
return false;
return true;
} public static void main(String[] args) { Map<HashCode_Equals, String> map = new HashMap<HashCode_Equals, String>(); HashCode_Equals obj1 = new HashCode_Equals();
obj1.a = 1;
obj1.b = 1;
map.put(obj1, "test"); HashCode_Equals obj2 = new HashCode_Equals();
obj2.a = 1;
obj2.b = 1; System.out.println("obj1.equals(obj2)=" + obj1.equals(obj2));
System.out.println("map.get(obj2)=" + map.get(obj2));
} }

执行结果:

obj1.equals(obj2)=true
map.get(obj2)=test


关于equals()方法不再多余介绍,重点篇幅是关于hashcode()的如何更好实现

《effective java》第9条:覆盖equals时总要覆盖hashcode

规范要求:相等的对象必须有相同的散列码。否则从map中当作key值得到的散列码是不同的,从而取到的value结果是不同的,上述的例子就是一个很好的证明

在不覆写的情况下,hashcode方法是JAVA默认生成的。从代码执行结果看obj1.equals(obj2)为true时,仍然hashcode不同,也就说JAVA默认的是不符合要求的,要求我们必须覆写。

关于hashcode,如果方法写的简单会造成hashcode的不断冲突,这样每个对象被映射到同一个散列桶中,使得散列表退化为链表。使得本应线性时间运行的程序变成了平方级在运行。

所以effecive java同时给了实现的参考

最新文章

  1. html学习第二天—— 第七章——CSS样式基本知识
  2. SQLSERVER 复制同一张表的递归结构
  3. aspnet超级链接 传递 当前页面 textbox值
  4. eclipse/intellij idea 远程调试hadoop 2.6.0
  5. io流对文件读写操作
  6. Unity学习疑问记录之layer问题
  7. input type=&quot;submit&quot; 和&quot;button&quot;有什么区别?
  8. Window驱动开发
  9. php正则常用表达式
  10. Ubuntu下安装android studio的时候,无法进入图形界面--/usr/lib/jdk1.8.0_60/jre/lib/i386/libawt_xawt.so: libXtst.so.6: 无法打开共享对象文件: 没有那个文件或目录
  11. javascript中的一些基本方法收藏
  12. 为实体类增加toJSON方法
  13. LKD: Chapter 8 Bottom Halves and Deferring Work
  14. virtualbox虚拟机NAT模式下不能连接外网
  15. eclipse的maven项目中找不到Maven Dependencies
  16. Educational Codeforces Round 53 (Rated for Div. 2)
  17. HAWQ配置之HDFS HA
  18. centos下查看端口占用情况,杀死进程
  19. db2 索引
  20. newCachedThreadPool使用案例

热门文章

  1. 搭建lnmp教程
  2. bzoj 4538: [Hnoi2016]网络
  3. ubuntu12.04destdrop删除不必要的软件
  4. Python数据分析工具
  5. Spring之DAO一
  6. TensorFlow 代码行统计
  7. 初学者福音——10个最佳APP开发入门在线学习网站
  8. C# 字符串的连接
  9. 细说MyEclipse调试
  10. Robot Framework 学习笔记(二)-------第一个脚本