面试官:请问 equals() 和 "==" 有什么区别?
应聘者:
-
equals()方法用来比较的是两个对象的内容是否相等,由于所有的类都是继承自java.lang.Object类的,所以适用于所有对象,如果没有对该方法进行覆盖的话,调用的仍然是Object类中的方法,而Object中的equals方法返回的却是==的判断;
-
"==" 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象。
equals()作用
equals() 的作用是用来判断两个对象是否相等。
equals() 定义在JDK的Object.java中。通过判断两个对象的地址是否相等(即,是否是同一个对象)来区分它们是否相等。源码如下:
public boolean equals(Object obj) {
return (this == obj);
}
既然Object.java中定义了equals()方法,这就意味着所有的Java类都实现了equals()方法,所有的类都可以通过equals()去比较两个对象是否相等。但是,我们已经说过,使用默认的“equals()”方法,等价于“==”方法。因此,我们通常会重写equals()方法:若两个对象的内容相等,则equals()方法返回true;否则,返回fasle。
下面根据"类是否覆盖equals()方法",将它分为2类。
-
若某个类没有覆盖equals()方法,当它的通过equals()比较两个对象时,实际上是比较两个对象是不是同一个对象。这时,等价于通过“==”去比较这两个对象。
-
我们可以覆盖类的equals()方法,来让equals()通过其它方式比较两个对象是否相等。通常的做法是:若两个对象的内容相等,则equals()方法返回true;否则,返回fasle。
下面,举例对上面的2种情况进行说明:
没有覆盖equals()方法的情况
public class EqualsTest {
public static void main(String[] args) {
// 新建2个相同内容的Person对象,
// 再用equals比较它们是否相等
User user1 = new User("James", 100);
User user2 = new User("James", 100);
System.out.printf("比较结果:" + user1.equals(user2));
}
/**
* @desc User类。
*/
static class User {
int age;
String name;
public User(String name, int age) {
this.name = name;
this.age = age;
}
public String toString() {
return name + " - " + age;
}
}
}
运行结果:
比较结果:false
结果分析:我们通过 user1.equals(user2) 来“比较user1和user2是否相等时”。实际上,调用的Object.java的equals()方法,即调用的 (user1==user2) 。它是比较“p1和p2是否是同一个对象”。而由 user1 和 user2 的定义可知,它们虽然内容相同;但它们是两个不同的对象,因此,返回结果是false。
覆盖equals()方法的情况
修改上面的EqualsTest,覆盖equals()方法:
public class EqualsTest {
public static void main(String[] args) {
// 新建2个相同内容的Person对象,
// 再用equals比较它们是否相等
User user1 = new User("James", 100);
User user2 = new User("James", 100);
System.out.printf("比较结果:" + user1.equals(user2));
}
/**
* @desc User类。
*/
static class User {
int age;
String name;
public User(String name, int age) {
this.name = name;
this.age = age;
}
public String toString() {
return name + " - " + age;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
User other = (User) obj;
if (age != other.age)
return false;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
return true;
}
}
}
运行结果:
比较结果:true
结果分析:我们在EqualsTest.java 中重写了User的equals()函数:当两个User对象的 name 和 age 都相等,则返回true。因此,运行结果返回true。
== 的作用
“==”:它的作用是判断两个对象的地址是不是相等。即判断引用对象是不是指向的堆中的同一个对象,我们知道,凡是new出来的对象都在堆中。而对象的引用都存放在栈中,具体来讲就是放在栈帧中,我们来看下面一段代码:
public static void main(String[] args) {
User user1 = new User("James", 100);
User user2 = new User("James", 100);
System.out.println("user1.equals(user2):" + user1.equals(user2));
System.out.println("user1==user2:" + (user1==user2));
}
输出结果:
user1.equals(user2):true
user1==user2:false
用内存图表示如下:
指向的是堆中两块不同的区域,所以用 "==" 比较时返回的是false。