Java中Integer和int的區(qū)別解讀
一、介紹
各位小伙伴們無論在工作還是學(xué)習(xí)中,與Integer都有著過硬的交情,我說的沒錯吧,大家都知道他可以表示一個整數(shù),而且也知道可以表示整數(shù)的還有int,只是使用Integer的次數(shù)要比int多得多,今天我們就來好好探究一下Integer與int的區(qū)別以及更深處的知識。
二、不同點(diǎn)
Integer是包裝類型(即引用類型),int是基本類型。
也就是說,Integer是一個類,里面有很多方法可以使用,例如
boolean equals(Object obj){...}
int compareTo(Integer anotherInteger){...}而int不是一個類,能做的只有+、-、*、/、=五種運(yùn)算,例如
int a = 3; int b = 4; int c = a + b;
Integer的默認(rèn)值是null,int的默認(rèn)值是0

Integer變量使用之前必須先實(shí)例化,int變量可以直接使用。
Integer如果沒有實(shí)例化就直接使用會報空指針異常(因默認(rèn)值為null);int如果沒有初始化則使用默認(rèn)值0。
三、相同點(diǎn)
取值范圍相同
最小值:Integer.MIN_VALUE= -2147483648 (-2的31次方)
最大值:Integer.MAX_VALUE= 2147483647 (2的31次方-1)

原因:眾所周知,整型變量在內(nèi)存中的字節(jié)數(shù)是4個字節(jié),一個字節(jié)是8位,因此一個整型變量占據(jù)32位,且首位為0表示非負(fù)數(shù),1表示負(fù)數(shù)。
四、使用equals()和==的區(qū)別
對于每個java開發(fā)人員來說,Integer的用法再熟悉不過了,但仍然會有些工作一兩年的小伙伴依然搞不清楚不同情況下equals()方法和 == 等號究竟有什么結(jié)果,以至于在基礎(chǔ)的面試過程中翻跟頭體驗(yàn)屢試不爽。這次我們一次性把所有可能以及結(jié)果出清楚。
equals()
此方法由Object類定義,在Integer類中又對其進(jìn)行了重寫,我們來看一下重寫后的方法。
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
這個方法先判斷參數(shù)對象是否繼承了Integer引用類型(如果參數(shù)是int基本類型,會先將其自動裝箱成引用類型),如果沒有繼承Integer引用類型,則返false,這個很好理解,如果數(shù)字和字符相比,那叫什么事嘛。如果繼承了Integer引用類型,就先把它強(qiáng)轉(zhuǎn)成Integer類型,再通過intValue()方法將其拆箱為int基本類型,再和value進(jìn)行地址比較(==右邊的value表示的是此Integer對象在常量池中的值,該值其實(shí)也是及本來想常量)。
? 說了一大堆,簡單一句話就是equals()方法進(jìn)行的是常量池中的地址是否相等。
? 另外一點(diǎn):常量池中值相等的數(shù)據(jù)不會同時出現(xiàn),常量池中只保存一份相同的值。
==
比較兩個對象在內(nèi)存中的地址是否相等。
下面是我們使用equals()方法和==方法來判斷Integer和int的區(qū)別。在此之前,我們要先知道裝箱和拆箱、常量池對象和堆內(nèi)存對象兩個概念:
裝箱
將基本數(shù)據(jù)類型變成引用類型的過程。發(fā)生在將基本數(shù)據(jù)類型賦值給引用類型的時候, 如Integer i = 5;
拆箱
將引用類型變成基本數(shù)據(jù)類型的過程。發(fā)生在將引用類型和基本數(shù)據(jù)類型做運(yùn)算的時候,如下所示
Integer i = 5; int j = 5; System.out.println(i==j);
常量池對象
保存的是創(chuàng)建對象時的字面量,相等的字面量只保存一次
堆內(nèi)存對象
保存的是new一個對象時這個對象在堆內(nèi)存中的地址
| 聲明變量 | 使用equals()方法比較 | 使用==比較 | |
| 1 | Integer i = new Integer(1); Integer j = new Integer(1); | true | false |
| 2 | Integer i = new Integer(1); Integer j = 1; | true | false |
| 3 | Integer i = new Integer(1); int j = 1; | true | true |
| 4 | Integer i = 1; Integer j = 1; | true | true |
| 5 | Integer i = 1; int j = 1; | true | true |
| 6 | int i = 1; int j = 1; | —— | true |
| 7 | Integer i = new Integer(128); Integer j = new Integer(128); | true | false |
| 8 | Integer i = new Integer(128); Integer j = 128; | true | false |
| 9 | Integer i = new Integer(128); int j = 128; | true | true |
| 10 | Integer i = 128; Integer j = 128; | true | false |
| 11 | Integer i = 128; int j = 128; | true | true |
| 12 | int i = 128; int j = 128; | —— | true |
情況1
在Integer i = new Integer(1)的過程中,java為我們做了兩件事,先在常量池中創(chuàng)建一個int=1的常量,再在堆內(nèi)存中創(chuàng)建一個對這個常量的地址引用,因此在用equals()方法時比較的是常量池中int=1這個常量,自己和自己比較的結(jié)果當(dāng)然為true;
而在==運(yùn)算時,直接比較了堆內(nèi)存中的地址,于是比較的結(jié)果為false
情況2
在Integer j = 1的過程中,實(shí)際上是java為我們在常量池中生成了一個常量對象1,然后j對象指向這個常量對象的地址。
equals()運(yùn)算結(jié)果為true是因?yàn)楸容^的是常量池中對象的比較,實(shí)際上常量池中只有一個值為1的對象,即自己比較自己。
==運(yùn)算結(jié)果為false是因?yàn)楸容^的是堆內(nèi)存中對象j的地址和常量池中常量對象1的地址,是兩個不同的地址比較
情況3
equals()運(yùn)算結(jié)果為true可以參考上面的源碼以及下方解釋
==運(yùn)算結(jié)果為true是因?yàn)橐妙愋虸nteger和基本數(shù)據(jù)類型int比較時,引用類型會自動拆包裝為int,然后進(jìn)行比較,實(shí)際上就變?yōu)閮蓚€int變量的比較,也是堆內(nèi)存中常量自己與自己的地址比較
情況4
變量i與變量j指向的其實(shí)都是常量池中的常量對象1,因此equals()運(yùn)算和==運(yùn)算的結(jié)果都是true
情況5
與情況4相同, 如果有疑惑可在文章后面查找答案
情況6
基本數(shù)據(jù)類型只能進(jìn)行加、減、乘、除、等于五種運(yùn)算,沒有equals()方法,==運(yùn)算為true還是因?yàn)槎褍?nèi)存中常量自己與自己進(jìn)行地址比較。
情況7
new Integer(128)會先在常量池中創(chuàng)建一個常量對象128,再在堆內(nèi)存中保存這個常量對象的地址,Integer i = new Integer(128)中變量i指向堆內(nèi)存中保存這個常量對象地址的地址,它是指向堆內(nèi)存的??梢岳斫鉃閮蓚€不同的堆內(nèi)存地址指向相同的常量池地址。
equals()運(yùn)算結(jié)果為true是因?yàn)楸容^的是常量池中的地址,而該地址只有一個,自己比較自己,結(jié)果為true
==運(yùn)算比較的是這兩個堆內(nèi)存的地址,因此結(jié)果為false
情況8
equals()運(yùn)算結(jié)果為true說了很多遍了,都是同一個原因
==運(yùn)算為false是因?yàn)楸容^的是堆內(nèi)存中兩個對象的地址,原因和情況5相同,可在文章后面查找答案
情況9
與情況3相同
情況10
與情況8相同,可在文章后面查找答案
情況11
與情況3相同
情況12
與情況6相同
五、解惑
JVM中一個字節(jié)一下的整型數(shù)據(jù)(即[128,127])會在JVM啟動時加載進(jìn)內(nèi)存,除非用new Integer()顯示的創(chuàng)建對象,否則都是同一對象。當(dāng)使用Integer i = 1時,會將1這個數(shù)字進(jìn)行緩存,下次再運(yùn)行Integer j = 1時,就會直接從緩存中取,就不會new了, 這樣取到的1 地址也是相同的, ==判斷返回true。
對于-128 ~ 127范圍之外的數(shù) 實(shí)際上就是new得到的, 地址不同==判斷為false
java在編譯Integer i = 100 ;時,java內(nèi)部會將其翻譯成為Integer i = Integer.valueOf(100);
所以關(guān)鍵就是看valueOf()函數(shù)了。JDK源碼的valueOf()是這樣的

到此,Integer與int運(yùn)算時的迷之操作總算解釋清楚了,如果有需要指正或者補(bǔ)充的,請小伙伴們掃下方二維碼關(guān)注公眾號并在后臺留言,我會及時給予回復(fù),讓我們一起把java學(xué)到荒,頭發(fā)掉到光。
到此這篇關(guān)于Java中Integer和int的區(qū)別解讀的文章就介紹到這了,更多相關(guān)Integer和int的區(qū)別內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring boot中使用RabbitMQ routing路由詳解
本篇文章主要介紹了spring boot中使用RabbitMQ routing路由詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-03-03
Java Map遍歷2種實(shí)現(xiàn)方法代碼實(shí)例
這篇文章主要介紹了Java Map遍歷2種實(shí)現(xiàn)方法代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10
SpringBoot 增量部署發(fā)布的實(shí)現(xiàn)步驟
本文介紹了通過拆分項目jar包和使用類加載器實(shí)現(xiàn)Spring Boot的增量部署,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-12-12
MyBatis-Plus流式查詢的實(shí)現(xiàn)示例
MyBatis-Plus 從 3.5.4 版本開始支持流式查詢,通過ResultHandler接口實(shí)現(xiàn)結(jié)果集的流式查詢,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-12-12
spring boot多數(shù)據(jù)源動態(tài)切換代碼實(shí)例
這篇文章主要介紹了spring boot多數(shù)據(jù)源動態(tài)切換代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-01-01
???????Spring多租戶數(shù)據(jù)源管理 AbstractRoutingDataSource
本文技術(shù)了???????Spring多租戶數(shù)據(jù)源管理 AbstractRoutingDataSource,下文詳細(xì)內(nèi)容介紹,需要的小伙伴可以參考一下2022-05-05

