`
xifangyuhui
  • 浏览: 186023 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

对java中equals和hashCode函数的一些理解1(转)

阅读更多

JDK的java.lang.Object类中实现了equals函数,其定义说明如下: 

其具体的实现代码如下所示: 

Java代码
  1. public boolean equals(Object obj) {  
  2.     return (this == obj);  
  3. }    


     从上面的代码中可以看出,Object类的equals实现只是简单地调用了“==”运算符,也即定义中说明的,对于两个非空对象X和Y,有且仅当X和Y指向同一个对象时该函数才返回true。由于Object类是java中所有类的超类,因而其它任何类都默认继承了此函数。在大多数情况下,此默认的实现方式是合适的,因为任一个类本质上都是唯一的;但是对于那些表示“值”的类(如Integer、String等),它们引入了自己特有的“逻辑相等”的概念,此时就必须修改equals函数的实现。 
例如java.lang.String类的实现如下: 

Java代码
  1.    public boolean equals(Object anObject) {  
  2.        if (this == anObject) {  
  3.     return true;  
  4. }  
  5. if (anObject instanceof String) {  
  6.     String anotherString = (String)anObject;  
  7.     int n = count;  
  8.     if (n == anotherString.count) {  
  9.     char v1[] = value;  
  10.     char v2[] = anotherString.value;  
  11.     int i = offset;  
  12.     int j = anotherString.offset;  
  13.     while (n-- != 0) {  
  14.         if (v1[i++] != v2[j++])  
  15.         return false;  
  16.     }  
  17.     return true;  
  18.     }  
  19. }  
  20. return false;  
  21.    }  


而java.lang.Integer类的实现如下: 

Java代码
  1. public boolean equals(Object obj) {  
  2.     if (obj instanceof Integer) {  
  3.         return value == ((Integer)obj).intValue();  
  4.     }  
  5.     return false;  
  6.     }  


      String类的equals实现使得当两个String对象所表示的字符串的值是一样的时候就能返回true;而Integer类则当两个Integer对象所持有的整数值是一样的时候就能返回true。 
      因此,当我们我们自己编写的类需要引入“逻辑相等”的概念,且其超类又没有改写equals的实现时,我们就必须实现自己的equals函数。在上面Object类中关于equals函数的说明中,定义了5条改写equals函数所必须遵守的规范。其中第1条自反性(reflexivity)和第5条通常能够自行满足。下面我们重点介绍下第2点对称性(symmetric)、第3点传递性(transitive)和第4点一致性(consistent)。 
      对称性 
即对于任意两个引用值X和Y,X.equals(Y) 和 Y.equals(X) 返回的结果必须是一致的。下面引用《effect java》上的例子来说明了违反这一特性的后果: 

Java代码
  1. public class CaseInsensitiveString {  
  2.       
  3.     private String s;  
  4.       
  5.     public boolean equals(Object obj){  
  6.         if(obj instanceof CaseInsensitiveString)  
  7.             return s.equalsIgnoreCase(((CaseInsensitiveString)obj).s);  
  8.         if(obj instanceof String)  
  9.             return s.equalsIgnoreCase((String)obj);  
  10.         return false;  
  11.     }  
  12. }  


然后我们定义如下的类来进行测试: 

Java代码
  1. public void test(){  
  2.     CaseInsensitiveString cis = new CaseInsensitiveString("Value");  
  3.     String s = "Value";  
  4.     List list = new ArrayList();  
  5.     list.add(cis);  
  6.     System.out.println(cis.equals(s));  
  7.     System.out.println(s.equals(cis));  
  8.     System.out.println(list.contains(s));  
  9. }  


运行test(),第一条语句的输出为true,而后两条为false。因为CaseInsensitiveString类中equals函数对String对象的比较也可能返回true,只要其大小写不敏感的字符串值相等;而在String类中对任何非String对象都返回false,这样CaseInsensitiveString的实现就违反了对称性,从而导致了第三条语句也返回了意想不到的结果false。这是因为ArrayList的contains函数实现是用实参s对list中每一个对象调用equals函数,若有equals函数返回true,则contains返回true;因而在这里contains函数必然返回false。因此,如果违反了对称性,则可能会得到意料之外的结果。解决此问题的一个办法就是在CaseInsensitiveString类的equals函数中对Stirng对象的比较无论值是否一致都返回false。 
      传递性 
即对于任意的引用值X、Y和Z,如果 X.equals(Y) 和 Y.equals(Z) 都返回true,则 X.equals(Z) 也一定返回true。下面举例说明: 

Java代码
  1.  public class Point {  
  2.     private final int x;  
  3.   
  4.     private final int y;  
  5.   
  6.     public Point(int x, int y) {  
  7.         this.x = x;  
  8.         this.y = y;  
  9.     }  
  10.   
  11.     public boolean equals(Object obj) {  
  12.         if (!(obj instanceof Point))  
  13.             return false;  
  14.         Point p = (Point) obj;  
  15.         return p.x == x && p.y == y;  
  16.     }  
  17. }  
  18.   
  19. public class ColorPoint extends Point {  
  20.   
  21.     private int z;  
  22.   
  23.     public ColorPoint(int x, int y, int z) {  
  24.         super(x, y);  
  25.         this.z = z;  
  26.     }  
  27.   
  28.     public boolean equals(Object obj) {  
  29.         if (!(obj instanceof ColorPoint))  
  30.             return false;  
  31.         ColorPoint cp = (ColorPoint) obj;  
  32.         return super.equals(obj) && cp.z == this.z;  
  33.     }  
  34. }  


在上面的代码中定义了类Point及其子类ColorPoint,并分别改写了equals函数。下面运行如下代码来测试: 

Java代码
  1. ColorPoint cp1 = new ColorPoint(1,2,3);  
  2. ColorPoint cp2 = new ColorPoint(1,2,4);  
  3. Point p = new Point(1,2);  
  4. System.out.println(cp1.equals(p));  
  5. System.out.println(p.equals(cp2));  
  6. System.out.println(cp1.equals(cp2));  


结果前两条语句输出“true”,而最后一条语句输出为“false”;从而违反了传递性规范。其主要原因是在基类的equals函数中并未对子类加以区分,从而使得基类与子类之间的比较也能返回true。要解决此方法,《effect java》上提供了“复合代替继承”的方法;另外,如果可以将基类和子类视为不等的话,使用 getClass()函数代替 instanceof 运算符进行比较可以很好的解决这一问题。getClass()会返回此对象的运行期类(runtime class),因为基类和子类属于不同的class,getClass()能够将其区分开来。下面是一段示例的代码: 

Java代码
  1. public boolean equals(Object obj) {  
  2.         if(obj.getClass() != this.getClass())  
  3.             return false;  
  4.         Point p = (Point) obj;  
  5.         return p.x == x && p.y == y;  
  6.           
  7.     }  


      

分享到:
评论

相关推荐

    equals 和 hashCode两者效果分析详解.docx

    原因是因为,在Java自带的容器HashMap和HashSet中, 都需同时要用到对象的hashCode()和equals()方法来进行判断,然后再插入删除元素,这点我们一会再谈。 那么我们还是单独来看hashCode(),为什么HashMap需要用到...

    探索equals()和hashCode()方法_动力节点Java学院整理

    equals():反映的是对象或变量具体的值,即两个对象里面包含的值--可能是对象的引用,也可能是值类型的值。 hashCode():计算出对象实例的哈希码,并返回哈希码,又称为散列函数。根类Object的hashCode()方法的...

    解析Java对象的equals()和hashCode()的使用

     在Java语言中,equals()和hashCode()两个函数的使用是紧密配合的,你要是自己设计其中一个,要设计另外一个。在多数情况 下,这两个函数是不用考虑的,直接使用它们的默认设计可以了。但是在一些情况下,这两个...

    Java synchronized详细解读.docx

    synchronized关键字可以作为函数的修饰符,也可作为函数内的语句,也就是平时说的同步方法和同步语句块。如果 再细的分类,synchronized可作用于instance变量、object reference(对象引用)、static函数和class ...

    Lombok(Java库)

    @EqualsAndHashCode:自动生成 equals 和 hashCode 方法,用于对象的比较和哈希处理。 @NoArgsConstructor / @RequiredArgsConstructor / @AllArgsConstructor:自动生成构造函数。 @Data:整合了 @Getter、@...

    AIC的Java课程1-6章

    第9章 常用类 4课时  理解Object类及其常用方法equals,hashCode和finalize等。  能够使用String,StringBuffer,StringBuilder类创建字符串对象和使用其方法,分辨不同类之间的区别。 ...

    kassava:该库提供了一些有用的kotlin扩展函数,用于在没有所有样板的情况下实现toString(),hashCode()和equals()

    卡萨瓦 该库提供了一些有用的kotlin扩展函数,用于在没有所有样板的情况下实现toString() , equals()和hashCode() 。 该库的主要动机是用于无法使用数据类且需要通过以下方式实现toString() / equals() / hashCode...

    Java基础知识点总结.docx

    equals()方法和hashCode()方法 270 数据结构 273 Array方法类汇总 304 Java数组与集合小结 305 递归 309 对象的序列化 310 Java两种线程类:Thread和Runnable 315 Java锁小结 321 java.util.concurrent.locks包下...

    写给大忙人看的JAVA SE 8

    9.3 实现equals、hashCode和compareTo方法 198 9.3.1 安全的Null值相等测试 198 9.3.2 计算哈希码 199 9.3.3 比较数值类型对象 200 9.4 安全需要 201 9.5 其他改动 204 9.5.1 将字符串转换为数字 204 9.5.2 全局...

    疯狂JAVA讲义

    第1章 Java概述 1 1.1 Java语言的发展简史 2 1.2 Java的竞争对手及各自优势 4 1.2.1 C#简介和优势 4 1.2.2 Ruby简介和优势 4 1.2.3 Python的简介和优势 5 1.3 Java程序运行机制 5 1.3.1 高级语言的运行机制 6...

    Java常见面试题208道.docx

    3.两个对象的 hashCode()相同,则 equals()也一定为 true,对吗? 4.final 在 java 中有什么作用? 5.java 中的 Math.round(-1.5) 等于多少? 6.String 属于基础的数据类型吗? 7.java 中操作字符串都有哪些类?它们...

    Java核心技术II(第8版)

    10.4.5 远程对象与equals、hashCode和clone方法 10.5 远程对象激活 10.6 Web Services与JAX-WS 10.6.1 使用JAX-WS 10.6.2 Web服务的客户端 10.6.3 Amazon的E-Commerce服务 第十一章 脚本、编译与注解处理 11.1 Java...

    java8集合源码分析-Outline:大纲

    equals和hashcode(, ) string,stringbuffer和stringbuilder(,,,, ) 伪泛型(, , ) 自动装箱(,) Try-with-resources() 序列化 反序列化(, , , ) interface 和 abstract class 区别 变长参数( ) 枚举详解(, , , , ) ...

    java视频教程(52课)下 异常 IO 线程 类集框架 开发工具之Eclipse 综合练习

    44_equals函数的作用.mp4 45_hashCode()与toString().mp4 46_开发工具之Eclipse(一).mp4 47_开发工具之Eclipse(二).mp4 48_开发工具之Eclipse(三).mp4 49_开发工具之Eclipse(四).mp4 50_综合练习(一).mp4 51_综合...

    去年秋招整理了这份后端开发的核心面试题(偏Java),顺利拿到了腾讯offer,决定把它贡献出来

    前言 2019年为了秋招,精挑细学了这一份面试题,并且只挑重点和难点,此次整理包括 Java、数据结构与算法、计算机网络、操作系统、数据库等。如果你能把这些面试题都...java中==和equals和hashCode的区别 int与integ

    Poko:一个Kotlin编译器插件,用于对公共API中的数据类施加额外的关注

    像普通的Kotlin数据类一样,您所要做的就是在类的构造函数中提供成员。 然后为其提供@Poko批注,并享受生成的toString , equals和hashCode 。 (将添加针对Java使用者的Builder类和针对Kotlin使用者的DSL初始化器...

    Java开发实战1200例(第1卷).(清华出版.李钟尉.陈丹丹).part3

    本书是第II卷,以开发人员在项目开发中经常遇到的问题和必须掌握的技术为中心,介绍了应用Java进行桌面程序开发各个方面的知识和技巧,主要包括Java语法与面向对象技术、Java高级应用、窗体与控件应用、文件操作...

Global site tag (gtag.js) - Google Analytics