1. equals和==的异同
首先equals方法是Object类的方法,任何类都默认继承了Object类。
共同点:都可以做比较,返回值都是boolean
区别:
- ==号是比较运算符,既可以比较基本数据类型,也可以比较引用数据类型,基本数据类型比较的是值,引用数据类型(即对象)比较的是地址值。
- equals方法只能比较的是引用数据类型, equals方法在没重写之前,比较的是地址值,底层依赖的是==号,但是比较地址值是没有意义的,我们需要重写equals方法比较对象中的属性值。因为在开发中我们通常比较的是对象中的属性值,我们认为相同属性是同一个对象,比如一个Name对象,有姓氏、名字的成员变量,我们认为如果他们的firstName和lastName相等就是同一个人(不考虑重名)。
2. 如何重写equals的问题
- 如果你重写equals方法,就一定要重写hashCode方法。hashCode方法也是Object的一个方法。
- String类已经重写了equals方法和hashCode方法。
可以这么理解,就是对象调用equals时,首先用hashCode()比较哈希值,然后再调用你自己重写的equals,相当于hashCode是隐式调用。
因此hashCode和equals有如下关系:
object1.equals(object2)为true是object1.hashCode()==object2.hashCode()为true的充分不必要条件!
当object1.hashCode()==object2.hashCode()为false时,object1.equals(object2)也为false。
当object1.hashCode()==object2.hashCode()为true时,object1.equals(object2)可能为true,也可能为false。
但是equals方法重写比较容易,就是比较他们的属性,hashCode方法如何重写呢?看如下代码:
public class Name {
private String firstName;
private String lastName;
public Name(String first, String last){
this.firstName = first;
this.lastName = last;
}
public String getFirstName() {
return this.firstName;
}
public boolean equals(Object obj) {
if(obj instanceof Name) {
Name name = (Name)obj;
return this.firstName.equals((name).firstName)
&&this.lastName.equals(name.lastName);
}
return super.equals(obj);
}
public int hashCode() {//hashcode的实现比较麻烦,用现成的String重写过的hashcode
return firstName.hashCode();
}
}
在这里,我们要知道,String类是已经重写了equals方法和hashCode方法,所以在我们自己的对象上,可以让我们对象的其中一个字符调用hashCode方法,即我们自己避免了重写hashCode方法。
3. 关于容器HashSet与equals
我们知道HashSet容器存储元素是无序不能重复的,所以我们要重写equals方法也要重写hashCode方法。如果我们不重写hashCode方法,任何对象都是可以存入的,因为一般情况下,哈希值是很难重复的,所以导致必然不equals,必然会存入Set中。
4. 解封装的扩展
- 基本型和基本型封装型进行“==”运算符的比较,基本型封装型将会自动拆箱变为基本型后再进行比较,因此Integer(0)会自动拆箱为int类型再进行比较,显然返回true;
- 两个Integer类型进行“==”比较,如果其值在-128至127,那么返回true,否则返回false, 这跟Integer.valueOf()的缓冲对象有关,这里不进行赘述。
- 两个基本型的封装型进行equals()比较,首先equals()会比较类型,如果类型相同,则继续比较值,如果值也相同,返回true
- 基本型的封装类型调用equals(),但是参数是基本类型,这时候,先会进行自动装箱,基本型转换为其封装类型,再进行3中的比较。
在jdk1.5之后,下列 java 程序输出结果为______。
int i=0;
Integer j = new Integer(0);
System.out.println(i==j);//true
System.out.println(j.equals(i));//true