總是遇到 equals 和 == 這類題,之前只知道對于基本數(shù)據(jù)類型, == 比較的是值匾竿,對于引用類型,== 比較的是內(nèi)存地址蔚万。關(guān)于 equals 方法岭妖,首先是在Object 中被定義的,它的定義中就是使用==方式來匹配的反璃。equals 默認是用來比較內(nèi)存地址的昵慌,但是像 String、Long淮蜈、Integer 等包裝類型斋攀,默認重寫 equals 方法,所以 equals 比較的就是值了礁芦。
下面對于此問題進行更詳細的解釋:
1.關(guān)于==
首先要知道==用于匹配內(nèi)存單元上的內(nèi)容蜻韭,其實就是一個數(shù)字悼尾,計算機內(nèi)部也只有數(shù)字,而在java語言中肖方,當==匹配時闺魏,就是比對兩個單元內(nèi)存的內(nèi)容是否一樣。
如果是原始類型俯画,byte,boolean,short,char,int,long,float,double,就是直接比較他們的值析桥。
如果是引用,比較的就是引用的值艰垂,引用的值可以被認為是對象的邏輯地址泡仗,如果兩個引用發(fā)生==操作,就是比較兩個相應(yīng)的對象的地址值是否一樣猜憎,換句話說娩怎,如果兩個引用保存的是同一個對象,則返回true,否則返回false胰柑。
2.關(guān)于equals()
equals方法截亦,首先是在Object中被定義的,它的定義中就是使用==方式來匹配的柬讨,也就是說崩瓤,如果不去重寫equals方法,并且對應(yīng)的類其父類列表中都沒有重寫過equals方法踩官,那么默認的equals操作就是對比對象的地址却桶。
equals方法之所以存在,是希望子類去重寫這個方法蔗牡,實現(xiàn)對比值的功能颖系,類似的,String就自己實現(xiàn)了該方法蛋逾。
int和int之間集晚,用==比較,肯定為true区匣。基本數(shù)據(jù)類型沒有equals方法
int和Integer比較蒋院,Integer會自動拆箱亏钩,== 和 equals都肯定為true
int和new Integer比較,Integer會自動拆箱欺旧,調(diào)用intValue方法, 所以 == 和 equals都肯定為true
Integer和Integer比較的時候姑丑,由于直接賦值的話會進行自動的裝箱。所以當值在
[-128,127]
中的時候辞友,由于值緩存在IntegerCache中栅哀,那么當賦值在這個區(qū)間的時候震肮,不會創(chuàng)建新的Integer對象,而是直接從緩存中獲取已經(jīng)創(chuàng)建好的Integer對象留拾。而當大于這個區(qū)間的時候戳晌,會直接new Integer。當Integer和Integer進行==比較的時候痴柔,在[-128,127]區(qū)間的時候沦偎,為true。不在這個區(qū)間咳蔚,則為false
當Integer和Integer進行equals比較的時候豪嚎,由于Integer的equals方法進行了重寫,比較的是內(nèi)容谈火,所以為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)建對象篷角,分配的地址不同,==比較的是內(nèi)存地址系任,所以肯定不同裝箱過程是通過調(diào)用包裝器的valueOf方法實現(xiàn)的
拆箱過程是通過調(diào)用包裝器的xxxValue方法實現(xiàn)的(xxx表示對應(yīng)的基本數(shù)據(jù)類型)
總結(jié):Byte恳蹲、Short、Integer俩滥、Long這幾個類的valueOf方法實現(xiàn)類似的嘉蕾。所以在[-128,127]區(qū)間內(nèi),==比較的時候霜旧,值總是相等的(指向的是同一對象)错忱,在這個區(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)) √