“==”和equals

一直对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)。

“==”和equals_第1张图片
图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_第2张图片
图2 Objects的equals方法源码
由图2可以看出Objects类的equals方法与关系操作符“==”其实是等价的。而String、Integer等类都对equals方法进行了重写,用来比较指向的对象所存储的内容是否相等。图3是String类equals方法的具体实现,可见equals的实现也是依靠关系操作符,只是将对象的内容彻底暴露了,绕开了地址值的比较。

“==”和equals_第3张图片
图3 String类的equals方法源码

另需注意:equals方法不能作用于基本数据类型的变量。

(三)总结

(1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;如果作用于引用类型的变量,则比较的是所指向的对象的地址。

(2)对于equals方法,如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址。诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。

(3)对于引用类型的变量,用“==”比较返回true的结果,那么用equals方法比较也一定返回true。



你可能感兴趣的:(“==”和equals)