網上搜了一遍,對于==和equals的表達感覺不全面;總感覺缺點什么;今天把這個比較總結出來三條規律。
結論1.基本類型沒有equals方法,只有==比較,比較的是值。
結論2.所有對象的==比較都是內存地址的比較
(上面的兩點簡單不介紹了)
首先我們看Integer類的比較。
1
2
3
4
|
Integer a= 1000 ; Integer b= 1000 ; System.out.println(a == b); //false System.out.println(a.equals(b)); //true |
因為a和b都是對象類型,所以都有自己的堆內存地址,所以根據結論2得出a==b是false。
至于equals我們看一下源碼
很明顯Integer的equals比較的是值。所以網上有些說法:equals比較的是內存地址的說法是以偏概全的;這個equals的比較要根據當前類的equals的實現。
所以a.equals(b)是值的比較。
1
2
3
4
|
Integer a1= 127 ; Integer a2= 127 ; System.out.println(a1 == a2); //true System.out.println(a1.equals(a2)); //true |
第三行a1==a2又變成了true;這個似乎違背了結論2.看看源碼吧
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
private static class IntegerCache { static final int low = - 128 ; static final int high; static final Integer cache[]; static { // high value may be configured by property int h = 127 ; String integerCacheHighPropValue = sun.misc.VM.getSavedProperty( "java.lang.Integer.IntegerCache.high" ); if (integerCacheHighPropValue != null ) { try { int i = parseInt(integerCacheHighPropValue); i = Math.max(i, 127 ); // Maximum array size is Integer.MAX_VALUE h = Math.min(i, Integer.MAX_VALUE - (-low) - 1 ); } catch ( NumberFormatException nfe) { // If the property cannot be parsed into an int, ignore it. } } high = h; cache = new Integer[(high - low) + 1 ]; int j = low; for ( int k = 0 ; k < cache.length; k++) cache[k] = new Integer(j++); // range [-128, 127] must be interned (JLS7 5.1.7) assert IntegerCache.high >= 127 ; } private IntegerCache() {} } |
Integer里面有一個靜態的IntergerCache類,里面有一個static靜態代碼塊和一個存放Integer的數組cache[].
上面代碼意思是:從jvm中取high值,如果有值的話和127比較取最大值,沒有值的話用127作為最大值。
-128作為最小值。所以cache[]數組的值是從-128~127并且是包裝類型。
回到上面a1==a2的問題為什么是true的問題。
因為a1和a2的值是127在整型的緩存里面,所以a1,a2指向的對象都是緩存里面的對象,所以a1==a2比較的依然是引用,只不過他們的引用都一樣而已。
如果超過緩存的范圍,就需要重新new了,就會出現新的對象,自然引用就不一樣了;所以a1==a2=1000的包裝類是false.
順便看了一下整型其他包裝類(Byte,Short,Long)的源碼,范圍都是-128~127,并且不可設置的。
Boolean同樣適合上面結論1和結論2.
Float的equals比較:
equals比較的是數值的二進制直接轉換成int型的值比較
Double的equals比較:
equals比較的是數值的二進制直接轉換成long型的值比較
至于我們自定義的類,比如定義一個Student類,其實它們都是調用Object的equals方法。
比較的是對象的引用,也就是內存地址的比較。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
package a; public class Student { static class A{ @Override public boolean equals(Object obj) { return true ; } } static class B{ } public static void main(String[] args) { A a= new A(); System.out.println(a.equals( new B())); } } |
上面對內部類的A方法重新了equals方法,總是返回true;那么傳入任何對象比較,都會是相等的。
結論3:自定義對象的equals比較方式取決于equals方法;如果沒有重寫,比較的就是引用;
如果進行了重寫,那么比較規則取決于equals體。
以上就是java中如何區分==和equals的詳細內容,更多關于JAVA ==和equals的資料請關注服務器之家其它相關文章!
原文鏈接:https://www.cnblogs.com/guoyansi19900907/p/12573157.html