Integer判断相等,到底该用==还是equals
概述
這幾天,被幾道java基礎(chǔ)練習(xí)題中的==和equals給繞暈了,所以打算把關(guān)于這塊的內(nèi)容好好總結(jié)下,爭取下次再遇到類似的題目,自己不會再出錯,恩,就是這樣。
簡單了解
在Object類中,equals方法的定義是這樣的,
public boolean equals(Object obj) { return (this == obj); }這塊就有點懵了,這不還是用==來進行比較的嗎?再往下搜了才知道,在大部分的封裝類中,都重寫了Object類的這個方法,所以兩者還是會有區(qū)別的。總的來說,==是一個關(guān)系運算符,如果比較的兩端都為基本類型,則判斷兩者的值是否相等,(判斷過程中還有不同基本類型的轉(zhuǎn)化,這里不做討論),如果比較的兩端都為引用類型的話,則比較兩者所指向?qū)ο蟮牡刂肥欠裣嗤?#xff1b;對于equals方法,首先,能調(diào)用這個方法肯定是一個對象,然后,如果這個對象所在的類重寫了equals方法,則按照重寫的方法進行比較,如果沒有,則比較兩者所指向?qū)ο蟮牡刂肥欠裣嗤?/p>
自己動手
了解了上面那么多內(nèi)容之后,然后自己就寫了一個小小的測試程序,打算驗證下,代碼如下:
public class Test{public static void main(String[] args) {Integer a = new Integer(200);Integer b = new Integer(200);Integer c = 200;Integer e = 200;int d = 200;System.out.println("兩個new出來的對象 ==判斷"+(a==b));System.out.println("兩個new出來的對象 equal判斷"+a.equals(b));System.out.println("new出的對象和用int賦值的Integer ==判斷"+(a==c));System.out.println("new出的對象和用int賦值的Integer equal判斷"+(a.equals(c)));System.out.println("兩個用int賦值的Integer ==判斷"+(c==e));System.out.println("兩個用int賦值的Integer equal判斷"+(c.equals(e)));System.out.println("基本類型和new出的對象 ==判斷"+(d==a));System.out.println("基本類型和new出的對象 equal判斷"+(a.equals(d)));System.out.println("基本類型和自動裝箱的對象 ==判斷"+(d==c));System.out.println("基本類型和自動裝箱的對象 equal判斷"+(c.equals(d)));} }執(zhí)行的結(jié)果如下
兩個new出來的對象 ==判斷false 兩個new出來的對象 equal判斷true new出的對象和用int賦值的Integer ==判斷false new出的對象和用int賦值的Integer equal判斷true 兩個用int賦值的Integer ==判斷false 兩個用int賦值的Integer equal判斷true 基本類型和new出的對象 ==判斷true 基本類型和new出的對象 equal判斷true 基本類型和自動裝箱的對象 ==判斷true 基本類型和自動裝箱的對象 equal判斷true首先,第9行中,對于兩個new出來的Integer對象,用==比較兩者,得到了false,這點應(yīng)該挺好理解的,每次使用new關(guān)鍵字,都會在堆內(nèi)存中申請一塊空間,存放相應(yīng)的對象的值,然后在棧中存放這塊內(nèi)存的引用。而==運算符比較兩者所指向?qū)ο蟮牡刂肥欠裣嗤?#xff0c;申請了兩塊空間,地址肯定不相同,所以結(jié)果為false。
第10行中,結(jié)果為true,查了下java的源碼,發(fā)現(xiàn)Integer重寫的equals方法是這樣的:
又調(diào)用了一個intValue方法,它的實現(xiàn)是這樣的:
public int intValue() { return value; }首先判斷傳進來的Object類型是不是Integer類的一個實例,如果不是直接返回false;如果是則判斷兩者的成員變量value值是不是相等的(Integer類中定義的private final int value),這塊又回到了基本類型的比較。value的值在創(chuàng)建這個對象的時候被賦值,兩個Integer對象傳遞的參數(shù)都為200,所以value值相等,返回true。
看第11行前,先看下第5行。第5行中,用int給Integer賦值的那條語句,從jdk1.5開始能這么做了,因為從這個版本開始,java引入了自動裝箱、自動拆箱機制。第5行就是一個自動裝箱的過程,相當(dāng)于:
Integer c = Integer.valueOf(200);
在Integer類中,valueOf方法是這么實現(xiàn)的:
上面這段代碼首先規(guī)定了一個范圍,默認是-128-127之間,如果參數(shù)中的i在這個范圍之內(nèi),則返回一個數(shù)組中的內(nèi)容,如果不在這個范圍,則new一個新的Integer對象并返回。查看Integer類的源碼可以發(fā)現(xiàn),這個數(shù)組里面緩存了基本類型-128-127之間的Integer對象。但是由于第11行是與一個new出來的對象做比較,所以==肯定返回的false。
第12行,equals方法比較兩個對象的value值,所以為true。
第13行,兩個自動裝箱的變量,但是裝箱傳遞的值大于127,所以返回false。這這塊又試了下在-128到127之間的數(shù),結(jié)果為true,大家可以試下。
第14行,結(jié)果為true。兩個自動裝箱的Integer對象,比較value。
第15行,這塊進行比較的時候,會對Integer對象進行自動拆箱,也就是調(diào)用intValue方法,方法如上。兩個基本數(shù)據(jù)類型進行==判斷,根據(jù)值比較,所以結(jié)果為true。這塊可能有人會問,為什么不是對int類型進行自動裝箱處理呢?其實這塊是java根據(jù)一個很明顯的道理進行設(shè)計的:如果有人比較一個int類型的值和Integer類型的值,是想比較什么呢?肯定是值呀,所以這塊是對Integer對象進行拆箱而不是對int類型裝箱了。
第16行這塊,首先調(diào)用equals方法的肯定是Integer對象,但是Integer類中重寫的equals方法參數(shù)是一個Object類型呀,怎么能傳遞一個基本數(shù)據(jù)類型進去呢?所以,這塊又是一個自動裝箱的表現(xiàn),當(dāng)傳遞一個int類型給equals這個方法時,java會自動將這個值打包裝箱為Integer類,而Integer類的最終父類又是Object,所以這塊參數(shù)的問題就解決了,然后就是兩個Integer對象進行equals判斷,返回true。
第17行,首先d為一個基本類型int,c為一個Integer對象,所以進行==比較的時候,肯定會對Integer對象進行拆箱處理,所以結(jié)果為true。
第18行,同第16行。
總結(jié)
“==比較地址,equals比較值”這個還是挺靠譜的。但是對于包裝類和基本類型,還要涉及它們的自動裝箱、自動拆箱,所以小心一點還是比較好的,不要走到別人挖的陷阱中。
作者:bu_想
鏈接:https://www.jianshu.com/p/9cb9c61b0986
來源:簡書
簡書著作權(quán)歸作者所有,任何形式的轉(zhuǎn)載都請聯(lián)系作者獲得授權(quán)并注明出處。
總結(jié)
以上是生活随笔為你收集整理的Integer判断相等,到底该用==还是equals的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Java 自定义 ClassLoader
- 下一篇: 小谈 Java 单元测试