前言
本篇文章講的是從JVM角度比較==和equals的區(qū)別
一:** Java數(shù)據(jù)類型分類**
1:基本數(shù)據(jù)類型
又稱為原始數(shù)據(jù)類型绍载,他們之間的比較應(yīng)該使用(==)诡宗,比較的是他們的值。
2:引用數(shù)據(jù)類型
當(dāng)引用數(shù)據(jù)類型用(==)進(jìn)行比較击儡,比較的是他們在內(nèi)存中的存放地址塔沃。
當(dāng)復(fù)合數(shù)據(jù)類型之間進(jìn)行equals比較時,這個方法的初始行為是比較對象在堆內(nèi)存中的地址曙痘。
equals()方法是用來判斷其他的對象是否和該對象相等.
//equals()方法在object類中定義如下:
public boolean equals(Object obj) {
return (this == obj);
}
但在一些諸如String,Integer,Date類中把Object中的這個方法覆蓋了,作用被覆蓋為比較內(nèi)容是否相同立肘。
// 比如在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;
}
}
很明顯边坤,這是進(jìn)行的內(nèi)容比較,而已經(jīng)不再是地址的比較谅年。依次類推Math茧痒、Integer、Double等這些類都是重寫了equals()方法的融蹂,從而進(jìn)行的是內(nèi)容的比較旺订。當(dāng)然弄企,基本類型是進(jìn)行值的比較。
二: String類的討論
String a = "abc";
String b = "abc";
System.out.println(a == b);//true
輸出true
說明:==在進(jìn)行復(fù)合數(shù)據(jù)類型比較時区拳,比較的是內(nèi)存中的存放地址拘领。因此a與b引用同一個String對象。
String b = "abc";
String c = new String("abc");
System.out.println(c == b);//false
System.out.println(c.equals(b));//true
輸出:
false
true
說明:b,c分別引用了兩個對象樱调。顯然约素,兩者內(nèi)容是相同的,因此equal返回true笆凌。第一個例子也一樣圣猎。
三: 解釋
String str1= "hello";
String str2= new String("hello");
String str3= str2;
首先看一張內(nèi)存上述的內(nèi)存分配圖
從圖中可以發(fā)現(xiàn)每個String對象的內(nèi)容實際是保存到堆內(nèi)存中的,而且堆中的內(nèi)容是相等的乞而,但是對于str1和str2來說所指向的地址堆內(nèi)存地址是不等的送悔,所以盡管內(nèi)容是相等的,但是地址值是不相等的
“==”是用來進(jìn)行數(shù)值比較的爪模,所以str1和str2比較不相等欠啤,因為str2和str3指向同一個內(nèi)存地址所以str2和str3是相等的。所以“==”是用來進(jìn)行地址值比較的呻右。
5. 為什么Java中1000==1000為false而100==100為true跪妥?
Integer i1 = 100, i2 = 100;
System.out.println(i1 == i2);//true
Integer i3 = 1000, i4 = 1000;
System.out.println(i3 == i4);//fales
查看Integer.java類,會發(fā)現(xiàn)有一個內(nèi)部私有類声滥,IntegerCache.java眉撵,它緩存了從-128到127之間的所有的整數(shù)對象。
看源碼
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer[] cache;
private IntegerCache() {
}
static {
int var0 = 127;
String var1 = VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
int var2;
if(var1 != null) {
try {
var2 = Integer.parseInt(var1);
var2 = Math.max(var2, 127);
var0 = Math.min(var2, 2147483518);
} catch (NumberFormatException var4) {
;
}
}
high = var0;
cache = new Integer[high - -128 + 1];
var2 = -128;
for(int var3 = 0; var3 < cache.length; ++var3) {
cache[var3] = new Integer(var2++);
}
assert high >= 127;
}
}
所以例子中i1和i2指向了一個對象落塑。因此100==100為true纽疟。
Demo地址
IT人生的Github地址:javaEquals
最后
不懂得地方歡迎私信我,我會在第一時間給予回復(fù)憾赁,如閱讀中發(fā)現(xiàn)寫錯的地方污朽,歡迎糾正。