總是遇到 equals 和 == 這類題,之前只知道對于基本數(shù)據(jù)類型, == 比較的是值,對于引用類型,== 比較的是內存地址。關于 equals 方法,首先是在Object 中被定義的,它的定義中就是使用==方式來匹配的。equals 默認是用來比較內存地址的,但是像 String、Long、Integer 等包裝類型,默認重寫 equals 方法,所以 equals 比較的就是值了。
下面對于此問題進行更詳細的解釋:
1.關于==
首先要知道==用于匹配內存單元上的內容,其實就是一個數(shù)字,計算機內部也只有數(shù)字,而在java語言中,當==匹配時,就是比對兩個單元內存的內容是否一樣。
如果是原始類型,byte,boolean,short,char,int,long,float,double,就是直接比較他們的值。
如果是引用,比較的就是引用的值,引用的值可以被認為是對象的邏輯地址,如果兩個引用發(fā)生==操作,就是比較兩個相應的對象的地址值是否一樣,換句話說,如果兩個引用保存的是同一個對象,則返回true,否則返回false。
2.關于equals()
equals方法,首先是在Object中被定義的,它的定義中就是使用==方式來匹配的,也就是說,如果不去重寫equals方法,并且對應的類其父類列表中都沒有重寫過equals方法,那么默認的equals操作就是對比對象的地址。
equals方法之所以存在,是希望子類去重寫這個方法,實現(xiàn)對比值的功能,類似的,String就自己實現(xiàn)了該方法。
int和int之間,用==比較,肯定為true。基本數(shù)據(jù)類型沒有equals方法
int和Integer比較,Integer會自動拆箱,== 和 equals都肯定為true
int和new Integer比較,Integer會自動拆箱,調用intValue方法, 所以 == 和 equals都肯定為true
Integer和Integer比較的時候,由于直接賦值的話會進行自動的裝箱。所以當值在
[-128,127]
中的時候,由于值緩存在IntegerCache中,那么當賦值在這個區(qū)間的時候,不會創(chuàng)建新的Integer對象,而是直接從緩存中獲取已經創(chuàng)建好的Integer對象。而當大于這個區(qū)間的時候,會直接new Integer。當Integer和Integer進行==比較的時候,在[-128,127]區(qū)間的時候,為true。不在這個區(qū)間,則為false
當Integer和Integer進行equals比較的時候,由于Integer的equals方法進行了重寫,比較的是內容,所以為true
Integer和new Integer : new Integer會創(chuàng)建對象,存儲在堆中。而Integer在[-128,127]中,從緩存中取,否則會new Integer.
所以 Integer和new Integer 進行==比較的話,肯定為false ; Integer和new Integer 進行equals比較的話,肯定為truenew Integer和new Integer進行==比較的時候,肯定為false ; 進行equals比較的時候,肯定為true
原因是new的時候,會在堆中創(chuàng)建對象,分配的地址不同,==比較的是內存地址,所以肯定不同裝箱過程是通過調用包裝器的valueOf方法實現(xiàn)的
拆箱過程是通過調用包裝器的xxxValue方法實現(xiàn)的(xxx表示對應的基本數(shù)據(jù)類型)
總結:Byte、Short、Integer、Long這幾個類的valueOf方法實現(xiàn)類似的。所以在[-128,127]區(qū)間內,==比較的時候,值總是相等的(指向的是同一對象),在這個區(qū)間外是不等的。
而Float和Double則不相等, Boolean的值總是相等的
測試題:
Integer a = 1000, b = 1000;
System.out.println(a == b);//1 false
Integer c = 100, d = 100;
System.out.println(c == d);//2 true
Integer s=new Integer(9);
Integer t=new Integer(9);
Long u=new Long(9);
- (s==u) ×
- (s==t) ×
- (s.equals(t)) √
- (s.equals(9)) √
- (s.equals(new Integer(9)) √