== 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象。比较的是真正意义上的指针操作。
equals用来比较的是两个对象的内容是否相等,由于所有的类都是继承自java.lang.Object类的,所以适用于所有对象,如果没有对该方法进行覆盖的话,调用的仍然是Object类中的方法,而Object中的equals方法返回的却是==的判断。
byte,short,char,int,long,float,double,boolean基本数据类型之间的比较需要用双等号(==),因为他们比较的是值。
接口、类、数组等非基本数据类型,Java中的字符串String属于引用数据类型。
当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。因为没new一次就会重新开辟一个新的堆内存空间。
Java中的==表示的是什么呢?有时候很令人费解。比如,以下例子输出是什么?
// 例一String str = "wo";String str1 = "wo";System.out.println("例一:" + (str == str1));// 例二String str2 = new String("wo");String str3 = new String("wo");System.out.println("例二:" + (str2 == str3));// 例三int i1 = 1;int i2 = 1;System.out.println("例三:" + (i1 == i2));// 例四TestObject t1 = new TestObject();TestObject t2 = new TestObject();System.out.println("例四:" + (t1 == t2));
最终结果如下:
例一:true例二:false例三:true例四:false
在Java中,如果是基本数据类型,则 == 比较的是值;如果是对象类型,则 == 比较的是对象的地址。但是,有时候会疑惑,String不是对象类型么?为什么例一是true呢?这个就要谈谈字符串常量池的问题。
String类是我们平常项目中使用频率非常高的一种对象类型,JVM为了提升性能和减少开销,避免字符串的重复创建,维护了一块特殊的内存空间,即字符串常量池。当需要使用字符串时,先去字符串常量池查看该字符串是否已经存在,如果存在,则可直接使用;如果不存在,初始化,并将该字符串放入到字符串常量池中。
「注意」:在JDK1.6及之前版本,字符串常量池在方法区中 在JDK1.7及以后版本,字符串常量池移到了堆中。
使用String str="wo",可能创建一个或者不创建对象。如果“wo”在字符串常量池中已经存在,则不会再创建String类型的值为“wo”的对象,而是将str指向这个“wo”对象内存地址,后续无论用这种方式创建多少个指向“wo”的引用,在内存中,都只有一个“wo”内存地址被分配。而==判断的是对象内存的地址,所以例一返回true。下图是用这种方式创建字符串的示例图。
图片
对象存放在堆中,字符串常量池是堆中一块特殊区域,new出来的是对象,字符串可以通过直接赋值创建一个对象,如上所述。对象的引用存放在栈中,String str是对象的引用。
在上图中,栈存放的是字符串的引用,str和str1存放的都是对象“wo”的内存地址,==判断对象时,判断的是他们存储的内存地址是否相同,由上图可见,他们的内存地址是相同的,所以例一输出的是true。
图片
例二的两个字符串都是通过new的方式创建对象的,所以在堆上有两个String对象,且这两个对象指向字符串常量池中的同一个对象“wo”,如上图所示,此时str2和str3存储的对象地址就不相同,所以例二返回的是false。
String str = new String("wo")创建了几个对象?如果字符串常量池中没有“wo”,则该句创建了两个对象,首先会创建一个“wo”存放在字符串常量池中,其本身就是一个对象;然后会new 一个字符串对象,并将“wo”的引用返回给new出来的对象;如果字符串常量池中有“wo”,则该句只创建了一个对象,因为该句首先会查找字符串常量池中是否存在“wo”,如果存在则直接返回"wo"的引用给new出来的对象。
Java中所有的类都是继承与Object这个基类的,在Object类中定义了一个equals方法,这个方法的初始行为是比较对象的内存地址,但在一些类库中已经重写了这个方法(一般都是用来比较对象的成员变量值是否相同),比如:String,Integer,Date 等类中,所以他们不再是比较类在堆中的地址。
public boolean equals(Object var1) { return this == var1;}
String类中重写后的代码:
public boolean equals(Object var1) { if (this == var1) { return true; } else { if (var1 instanceof String) { String var2 = (String)var1; int var3 = this.value.length; if (var3 == var2.value.length) { char[] var4 = this.value; char[] var5 = var2.value; for(int var6 = 0; var3-- != 0; ++var6) { if (var4[var6] != var5[var6]) { return false; } } return true; } } return false; }}
示例:
// abc在常量池中String a = "abc";// 栈中b指向常量池中的abcString b = "abc";// 在堆内存中重新开辟了一个abc的空间String c = new String("abc");//检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会 把"abc"添加到字符串池中,然后再返回它的引用。String d = c.intern(); System.out.println(a==b);System.out.println(a.equals(b));System.out.println(a==c);System.out.println(a.equals(c));System.out.println(a==d);System.out.println(a.equals(d));
运行结果:
truetruefalsetruetruetrue
引用类型:默认情况下,比较的是地址值,重写该方法后比较对象的成员变量值是否相同。
对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的实现来进行比较。
本文链接:http://www.28at.com/showinfo-26-13283-0.htmlJava中equals和==的区别
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com
上一篇: 通过线程池方式改造Stream.parallel()并行流
下一篇: 面试中如何答好:synchronized