美文网首页
“==”和equals

“==”和equals

作者: PingPi | 来源:发表于2017-03-20 00:40 被阅读0次

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

图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方法的作用,看源码是个比较好的方法。

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

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

(三)总结

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

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

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


相关文章

  • Java 比较相等

    Java Equals() 特性 Equals() 和 == 的区别 重写Equals方法 重写HashCode方法

  • equals,==和 equals,hashCode

    == : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象(基本数据类型==比较的是值,引...

  • Object的基本方法

    一 equals和hashCode equals和== equals的默认实现就是==,既比较两个对象的引用。 最...

  • 【面试1】

    基础 1、equals和==(equals和==的区别)equals:是Object的一个方法,实现对象的比较;复...

  • java容器

    equals和hashCode equals和hashcode间的关系:如果两个对象相同(即equals比较返回t...

  • Java

    equals和hashcode public boolean equals(Object obj){ retur...

  • ==和equals

    在初学Java时,可能会经常碰到下面的代码: 1String str1 =newString("hello");2...

  • ==和equals

    ==和equals的区别: ==和equals方法都是用来比较两个变量/对象是否相等。 如果没有重写equals方...

  • == 和 equals

    这个问题缠绕了很久。什么时候用==?基本数据类型时使用==,byte - long -int -short -ch...

  • ==和equals

    ==: ==: 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是是否相同,即是否是指的...

网友评论

      本文标题:“==”和equals

      本文链接:https://www.haomeiwen.com/subject/pydvnttx.html