Java中equals和==的区别
一.背景介绍
== 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象。比较的是真正意义上的指针操作。
equals用来比较的是两个对象的内容是否相等,由于所有的类都是继承自java.lang.Object类的,所以适用于所有对象,如果没有对该方法进行覆盖的话,调用的仍然是Object类中的方法,而Object中的equals方法返回的却是==的判断。
java中的数据类型可以分为两类:
基本数据类型
byte,short,char,int,long,float,double,boolean
基本数据类型之间的比较需要用双等号(==),因为他们比较的是值
引用数据类型
接口、类、数组等非基本数据类型
Java中的字符串String属于引用数据类型。因为String是一个类
当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。因为没new一次就会重新开辟一个新的堆内存空间
二.Java中的==
2.1Java中的==
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));
TestObject为一个类
public class TestObject {
public TestObject(){
}
}
最终结果如下:
例一:true
例二:false
例三:true
例四:false
其实,在Java中,如果是基本数据类型,则 == 比较的是值;如果是对象类型,则 == 比较的是对象的地址。但是,有时候会疑惑,String不是对象类型么?为什么例一是true呢?这个就要谈谈字符串常量池的问题。
总结:
- 如果是基本数据类型,==判断的是值 如果是对象类型,==判断的是对象的地址
通过直接赋值而不是new的方式给String赋值,如果字符串常量池中有该对象,则不会再创建,此时通过 ==
判断,返回的是true。如:String str=“wo”;String str1=“wo”;str == str1为true.
在JDK1.6及以前版本,字符串常量池在方法区中;在JDK1.7及以后,字符串常量池在堆中。 对象的引用保存在栈中。
三.java中equals()方法
Java中所有的类都是继承与Object这个基类的,在Object类中定义了一个equals方法,这个方法的初始行为是比较对象的内存地址,但在一些类库中已经重写了这个方法(一般都是用来比较对象的成员变量值是否相同),比如:String,Integer,Date 等类中,所以他们不再是比较类在堆中的地址了、
Object类中源码
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;
}
}
String类的equals()方法
String a = "abc";// abc在常量池中
String b = "abc";//栈中b指向常量池中的abc
String c = new String("abc");// 在堆内存中重新开辟了一个abc的空间
String d = c.intern();//检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会 把"abc"添加到字符串池中,然后再返回它的引用。
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));
结果:
true
true
false
true
true
true
equals 的作用:
引用类型:默认情况下,比较的是地址值,重写该方法后比较对象的成员变量值是否相同。
总结:
对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的要求来。