参考文章
前置知识
1.==
- 基本数据类型:比较的是值
- 引用数据类型:比较的是对象的地址(准确地说是这个对象的堆内存地址)
// 基本数据类型
int num1 = 666;
int num2 = 666;
System.out.println(num1 == num2); // true
// 引用数据类型
Student student1 = new Student().setName("zs");
Student student2 = new Student().setName("zs");
System.out.println(student1 == student2) // false
特别的:包装类型属于引用类型,除了Float和Double以外都有常量池。
以Integer为例,它的常量池范围为[-128, 127],值为常量池范围内的对象是会被复用的。观察下列代码:
- num1和num2的值都在常量池范围内,且值均为120。此时num1和num2共用一个Integer常量池中的对象,因而使用==判断,二者是相等的。
- num3和num4的值为128,不在[-128, 127]范围内,因此会重新创建对象。
- num5和num6的值虽然都为111且在常量池范围内,但num6用new Integer(111)创建了一个全新的对象,该对象与常量池对象的地址不一,因而不等。
2.Object.equals
Java中所有的类都继承与Object,其下的equals方法源码如下:
public boolean equals(Object obj) {
// this:调用者
// obj:传入的对象
return (this == obj);
}
如果不重写该方法,equals()就与==无异,因此在许多类被创建时,都应该重写其中的equals方法(也应重写hashCode()方法,详见[11.toString( )、hashCode( )和equals( ),Objects工具包](11.toString( )、hashCode( )和equals( ),Objects工具包.md))
2.1.包装类及String重写equals
Integer等包装类以及String类型都重写了equals方法,源码如下:
// Integer类重写equals
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
// String类重写equals
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
思想都是一样的:
- (String:判断地址是否一致,一致返回true,否则进入下一步)
- 判断类型是否一致,不一致则返回false,否则进入下一步
- 转换成基本数据类型,比较值是否相等
2.2.自定义类重写equals
通常情况下,自定义类判等的条件应该是:成员变量完全一致
因此在创建自定义类时,需要重写equals():
// Student类中有成员变量name和sex
// IDEA自动生成的JDK1.7重写模板
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student student = (Student) o;
return Objects.equals(name, student.name) &&
Objects.equals(sex, student.sex);
}
注意:由于Set、HashMap等在进行判等时,会调用Object.hashCode()判断哈希值是否一致,因此通常情况下还应重写hashCode()
详见[11.toString( )、hashCode( )和equals( ),Objects工具包](11.toString( )、hashCode( )和equals( ),Objects工具包.md)
// Student类中有成员变量name和sex
// IDEA自动生成的JDK1.7重写模板
@Override
public int hashCode() {
// 仅以传入的参数值计算hash,而不考虑对象对应的堆内存地址
return Objects.hash(name, sex);
}