一直对Java中的“==”和equals的用法一知半解,在实际使用中又因为Character、String等分装类且伴随toString和String.valueOf等方法的使用后则更加傻傻分不清楚,需要不断调试才能完全搞定,因此痛定思痛,收集了一串资料终于彻底搞懂后,写以此文,便于自身回顾,同时以飨读者。
个人认为《浅谈Java中的equals和“==”》 这篇文章讲的比较好,本文中的一些示例代码和关键知识点也引用于该文章。
要理解关系操作符“==”和equals的区别首先要搞清楚数据变量类型,JAVA中存在基本数据类型变量和非基本数据类型变量(专业点的叫法是对象的引用变量),Java中有8种基本数据类型:
浮点型:float(4 byte), double(8 byte)
整型:byte(1 byte), short(2 byte), int(4 byte) , long(8 byte)
字符型: char(2 byte)
布尔型: boolean
需要指出的是这8种基本数据类型的变量直接存放在栈中,变量直接存储的是“值”本身,而不是“地址”。因此在用关系操作符“==”来进行比较时,比较的就是 “值” (这样设计的原因是栈有一个很重要的特性,存取速度比堆要快,存在栈中的数据可以共享,说白了就是为了提升性能)。引用类型的变量存储的并不是对象的 “值”本身,而是于其关联的对象在内存中的地址(如图 1)。
看如下代码:
public class Test1 {
public static void main(String[] args) {
int n=3;
int m=3;
System.out.println(n==m); //true
String str = new String("abc");
String str1 = new String("abc");
String str2 = new String("abc");
System.out.println(str1==str2); //false
str1 = str;
str2 = str;
System.out.println(str1==str2); // true
}
由于m和n中直接存储的是值3,因此比较m==n时,等价于比较3==3?显然是相等的,返回true。而比较引用类型变量str1==str2时,等价于比较对象的地址值0x001==0x333?显然是不相等的,返回false,口语化的表示方式为str1和str2指向的是两个不同的对象。而将str引用对象的地址值赋值给变量str1和str2后,str1和str2都指向了堆中同一对象。
我们再来看一段代码:
public class Test1 {
public static void main(String[] args) {
System.out.println("abc"=="abc"); //true
System.out.println(new String("abc")==new String("abc")); // false
}
看了上面这段代码,估计很多人不敢相信这样的结果,那么就在电脑上自己敲一遍。要理解上述结果的差异,我们需要知道String是一种特殊的包装类数据类型,它不属于8种基本数据类型,但是我们可以像基本数据类型那样,不通过new的方法在栈中直接创建String类型数据,且也是共享的,意思是栈中只存在一个“abc”,因此比较的两个“abc”实际上是同一个。
而通过new()方法创建的两个String类型对象存放于堆中,虽然在栈中没有创建String类型的引用变量指向他们,但是还是比较的是两个对象的地址值。提示:只有通过new()方法才能保证每次都创建一个新的对象。
(二) equals方法
那如何才能判断两个引用类型变量所指向对象的内容是否相等呢?这时候该equals上场了。equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。要理解equals方法的作用,看源码是个比较好的方法。
另需注意:equals方法不能作用于基本数据类型的变量。
(三)总结
(1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;如果作用于引用类型的变量,则比较的是所指向的对象的地址。
(2)对于equals方法,如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址。诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。
(3)对于引用类型的变量,用“==”比较返回true的结果,那么用equals方法比较也一定返回true。