目录
1.equals()的作用是什么?
定义在Object.java中的它,是根据判断两个对象之间的地址来判断两个对象是否相等,即等同于“==”。但我们通常会对它进行重写。
1.1重写equals()方法
代码:
package com.hbk.util;
/**
* @ProjectName: wxhospital
* @Package: com.hbk.util
* @ClassName: EqualsDemo1
* @Author: Sarah0429
* @Description: 重写equals(),用来比较值是否相等
* @Date: 2020/04/28 14:10
* @Version: 1.0
*/
public class EqualsDemo1 {
public static void main(String[] args) {
Person puyuma1 = new Person("puyuma", 1);
Person puyuma2 = new Person("puyuma", 1);
System.out.println(puyuma1.equals(puyuma2));
}
private static class Person {
String name;
int sex;
public Person(String name, int sex) {
this.name = name;
this.sex = sex;
}
public String toString() {
return name + " - " + sex;
}
/**
* 重写equals()
*/
@Override
public boolean equals(Object obj) {
//判断传入对象是否为空
if (obj == null) {
return false;
}
//根据两个对象的地址判断是否为同一对象
if (this == obj) {
return true;
}
//判断两个对象的类型是否相同
if (this.getClass() != obj.getClass()) {
return false;
}
Person person = (Person) obj;
return name.equals(person.name) && sex == person.sex;
}
}
}
控制台打印:
由上可以看出,当我们重写了equals()时,它的作用是:当两个person的name与sex属性值相同时,返回true。
1.2不重写equals()方法
代码:
package com.hbk.util;
/**
* @ProjectName: wxhospital
* @Package: com.hbk.util
* @ClassName: EqualsDemo1
* @Author: Sarah0429
* @Description: 重写equals(),用来比较值是否相等
* @Date: 2020/04/28 14:10
* @Version: 1.0
*/
public class EqualsDemo1 {
public static void main(String[] args) {
Person puyuma1 = new Person("puyuma", 1);
Person puyuma2 = new Person("puyuma", 1);
System.out.println(puyuma1.equals(puyuma2));
}
private static class Person {
String name;
int sex;
public Person(String name, int sex) {
this.name = name;
this.sex = sex;
}
public String toString() {
return name + " - " + sex;
}
}
}
控制台打印:
由上可以看出,当我们未重写equals()时,它调用的是==,即判断是否为同一对象,而不是比较里面的具体内容。
1.3对于equals(),我们必须遵循如下规则:
- 对称性:如果x.equals(y)返回是"true",那么y.equals(x)也应该返回是"true"。
- 反射性:x.equals(x)必须返回是"true"。
- 类推性:如果x.equals(y)返回是"true",而且y.equals(z)返回是"true",那么z.equals(x)也应该返回是"true"。
- 一致性:如果x.equals(y)返回是"true",只要x和y内容一直不变,不管你重复x.equals(y)多少次,返回都是"true"。
- 非空性,x.equals(null),永远返回是"false";x.equals(和x不同类型的对象)永远返回是"false"。
2.equals() 与 == 的区别
看完上面的内容,相信大家对equals()也有了一个初步的理解。
equals()它一般有两种使用情况:
- 重写了equals(),我们一般都会重写equals(),用来判断两个对象之间的属性值是否相等
- 未重写equals(),即调用Object.java里的equals(),等同于“==”
3.hashCode() 的作用是什么?
它的作用是获取哈希码,也称为散列码,是一个int整数。这个哈希码的作用是确定该对象在哈希表中的索引位置。
hashCode() 定义在JDK的Object.java中,这就意味着Java中的任何类都包含有hashCode() 函数。
虽然,每个Java类都包含hashCode() 函数。但是,仅仅当创建并某个“类的散列表”(关于“散列表”见下面说明)时,该类的hashCode() 才有用(作用是:确定该类的每一个对象在散列表中的位置;其它情况下(例如,创建类的单个对象,或者创建类的对象数组等等),类的hashCode() 没有作用。
上面的散列表,指的是:Java集合中本质是散列表的类,如HashMap,Hashtable,HashSet。
也就是说:hashCode() 在散列表中才有用,在其它情况下没用。 在散列表中hashCode() 的作用是获取对象的散列码,进而确定该对象在散列表中的位置。
OK!至此,我们搞清楚了:hashCode()的作用是获取散列码。但是,散列码是用来干什么的呢?为什么散列表需要散列码呢?要解决这些问题,就需要理解散列表!关于散列表的内容,非三言两语道的明白;大家可以通过下面几篇文章来学习:
为了能理解后面的内容,这里简单的介绍一下散列码的作用。
我们都知道,散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码!
散列表的本质是通过数组实现的。当我们要获取散列表中的某个“值”时,实际上是要获取数组中的某个位置的元素。而数组的位置,就是通过“键”来获取的;更进一步说,数组的位置,是通过“键”对应的散列码计算得到的。
下面,我们以HashSet为例,来深入说明hashCode()的作用。
假设,HashSet中已经有1000个元素。当插入第1001个元素时,需要怎么处理?因为HashSet是Set集合,它不允许有重复元素。
“将第1001个元素逐个的和前面1000个元素进行比较”?显然,这个效率是相当低下的。散列表很好的解决了这个问题,它根据元素的散列码计算出元素在散列表中的位置,然后将元素插入该位置即可。对于相同的元素,自然是只保存了一个。
由此可知,若两个元素相等,它们的散列码一定相等;但反过来确不一定。在散列表中,
1、如果两个对象相等,那么它们的hashCode()值一定要相同;
2、如果两个对象hashCode()相等,它们并不一定相等。
注意:这是在散列表中的情况。在非散列表中一定如此!
3.1对于hashCode(),我们应该遵循如下规则:
1.在一个应用程序执行期间,如果一个对象的equals方法做比较所用到的信息没有被修改的话,则对该对象调用hashCode方法多次,它必须始终如一地返回同一个整数。
2.如果两个对象根据equals(Object o)方法是相等的,则调用这两个对象中任一对象的hashCode方法必须产生相同的整数结果。
3.如果两个对象根据equals(Object o)方法是不相等的,则调用这两个对象中任一个对象的hashCode方法,不要求产生不同的整数结果。但如果能不同,则可能提高散列表的性能。
4.两者之间的关系
两句话概括:
- 如果equals()返回“true”,那么两个对象的hashCode()必须相等。
- 如果equals()返回“false”,那么两个对象的hashCode()有可能相等,也有可能不等。