Java中的equals()和hashCode()

Java中有这么句话,如果重写了equals(),那么必须重写hashcode()。那么为啥要这么干呢?他俩到底是啥关系?怎样重写都行吗?今天来撸一撸经常用到的equals和hashcode()

Object 类

Object类是Java中最顶层的类,所有的类都继承自它。Object类中方法有:

1.clone方法 默认实现浅拷贝
2.getClass 方法 获得运行时类
3.toString 方法 将对象转化为字符串
4.finalize 方法 用于释放类所持有的资源 GC时才会调用
5.equals 方法 判断对象是否相同
6.hashCode 方法 返回对象的hash值
7.wait方法 线程等待
8.notify方法 唤醒等待线程
9.notifyAll方法 唤醒所有等待线程

equals()和hashCode()是属于Object的方法。默认equals采用类==号来判断,比较的是地址值,即判断两个引用是不是指向同一个对象。hashCode()计算方法随着JDK版本的不同有所区别,在JDK8中使用的一个和当前线程有关的随机值+3个确定值,用随机数算法对这4个值运算后得到。可以在JVM启动时选择默认的hashcode()算法。hashCode可选择算法如下:

第一种:随机数,返回一个Park-Miller伪随机数生成器生成的随机数。Java 6、Java 7 采用此种
第二种:将对象的内存地址值做移位运算后与一个随机数进行异或得到结果
第三种:默认返回1
第四种:返回一个值,相当于全局变量 每新增一个对象数值+1
第五种:返回对象的内存地址
第六种:采用一个和当前线程有关的随机值以及3个固定值,这四个值通过xorshift随机数算法得到的一个随机数,Java8采用此种。

Hash值相同,对象不一定相等,但Hash值不相等,则对象一定不相等。

在比较对象时随着场景的不同,比较多方法也不同。有些情况需要判断地址是否相同,有些需要判断对象的含义是否相同。比如Integer类,当Integer值在-128与127之间时,会指向同一个地址,但是当Integer的数值不在此区间,就会开辟一个空间来存放它,此时每个Integer所指向的地址不同。但从数值逻辑上判断,两个Integer(1000)值相同,默认的equals方法已经不能满足要求,所以就需要重写equals方法。所以很多类都重写了equals方法。

    Integer a = new Integer(1000);
    Integer b = new Integer(1000);
    System.out.println( a == b);
    System.out.println(a.equals(b));
	
	false //a b的地址不同
	true // a b的值相同

Java中要保证Hash值相同,对象不一定相等,但Hash值不相等,则对象一定不相等。基于此,重写了euqals就必须重写hashCode。因为默认情况下equals采用的是==,比较的是地址值,此时如果equals相同,那么引用的必定是同一个对象,同一个对象的Hash值肯定相同。而假设Hash值相同,但可能对象并不是用一个,equals并不相等。满足Hash值相同,对象不一定相等,但Hash值不相等,则对象一定不相等。

但如果只重写equals,比如上面的Integer。equals判断两个值相同,但是两个对象并不是同一个对象,Hash值必定不相同。这种情况下就是Hash值不等,对象相等,违反了原则。所以重写equals,就需要重写hashCode。更准确的说是怎样重写equals,就依照规则怎样重写hashCode。比如Integer的euqals,比较具体数值,那么HashCode也应该是基于数值的。equals要比较对象的ID,那么HashCode应该也是基于对象的ID。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Java equals() 和 hashCode() 是两个重要的方法,用于判断对象是否相等和哈希的计算。 equals() 方法用于比较两个对象是否相等。默认情况下,它使用 == 运算符比较对象的引用,即比较两个对象是否指向同一个内存地址。如果想比较两个对象的属性是否相等,就需要重写 equals() 方法,并根据对象的属性进行比较hashCode() 方法用于计算对象的哈希,即将对象映射为一个整数。哈希Java 经常用于散列表等数据结构,用于快速查找和比较对象。如果两个对象相等,它们的哈希应该相同。因此,重写 equals() 方法的同时,也需要重写 hashCode() 方法。 在重写 equals() 和 hashCode() 方法时,需要遵循一些规则。比如,如果两个对象相等,它们的 hashCode() 方法应该返回相同的;反之,如果两个对象的 hashCode() 相等,它们不一定相等,还需要通过 equals() 方法进行比较。此外,hashCode() 方法不能依赖于对象的内存地址或时间戳等不稳定因素,应该根据对象的属性计算哈希。 ### 回答2: JavaequalshashCode是非常重要的两个方法,在Java几乎所有的类都会覆盖它们,而且它们是非常重要的类比较和哈希计算的方法。 equals方法用于比较两个对象是否相等,通常比较两个对象的内容是否相同,他们是否具有相同的属性和equals方法的默认实现是比较两个对象的引用是否相同,即两个对象是否是同一个对象,不过这个默认的实现并不能满足所有的需求,因为有时候即使两个对象的引用不同,但他们的内容依然相同,所以需要重写equals方法来实现内容的比较。 另一方面,hashCode方法则是将一个对象映射成一个整型的哈希,通常用于快速查找或比较对象。 在Java,哈希表是非常常见的数据结构,而哈希就是在哈希表用来寻找和比较对象的唯一标识,所以hashCode方法的实现对于哈希表的性能有着非常重要的影响。 需要注意的是,hashCode方法必须和equals方法保持一致性,也就是说如果两个对象的equals方法返回true,那么他们的hashCode方法返回的哈希必须相同,反之亦然。 在实现equals方法时,一般会遵循一些基本原则,例如对称性,传递性,一致性等,同时还需要注意适当的判断null和使用instanceof进行类型判断。在实现hashCode方法时,需要确保相同的对象始终返回相同的哈希,同时也需要尽可能的让不同的对象返回不同的哈希。 总之,JavaequalshashCode方法是非常重要的两个方法,深入理解其实现原理和使用方式,可以有效提高Java应用程序的性能和稳定性。 ### 回答3: Javaequals()和hashCode()是两个非常重要的方法,它们用于判断对象之间的相等性和排序性。 equals()方法是用来比较两个对象是否相等的,它的作用是比较两个对象的内容是否相等,而不是比较两个对象引用是否相等。在默认情况下,equals()方法会比较两个对象的引用,即判断两个对象是否指向同一个内存地址。但是,我们通常需要自己来重写equals()方法,以便在比较对象时只比较对象的内容。 hashCode()方法是Java的哈希函数,它将对象映射到一个整数。这个整数通常用于将对象存储到哈希表,可以快速地定位对象。如果两个对象相等,那么它们的hashCode()方法返回的整数也应该相等。因此,在重写equals()方法时,我们也需要重写hashCode()方法。 在Java,如果两个对象调用equals()方法返回true,那么它们的hashCode()方法返回的整数也必须相等。因此,重写hashCode()方法时,必须保证相等的对象返回相等的hashCode(),否则将会影响哈希表的性能。同时,hashCode()方法的重写也应该考虑到对象的内容,以便产生尽可能不同的哈希
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值