Java中Integer和int的區(qū)別解讀
一、介紹
各位小伙伴們無(wú)論在工作還是學(xué)習(xí)中,與Integer都有著過(guò)硬的交情,我說(shuō)的沒(méi)錯(cuò)吧,大家都知道他可以表示一個(gè)整數(shù),而且也知道可以表示整數(shù)的還有int,只是使用Integer的次數(shù)要比int多得多,今天我們就來(lái)好好探究一下Integer與int的區(qū)別以及更深處的知識(shí)。
二、不同點(diǎn)
Integer是包裝類(lèi)型(即引用類(lèi)型),int是基本類(lèi)型。
也就是說(shuō),Integer是一個(gè)類(lèi),里面有很多方法可以使用,例如
boolean equals(Object obj){...} int compareTo(Integer anotherInteger){...}
而int不是一個(gè)類(lèi),能做的只有+、-、*、/、=五種運(yùn)算,例如
int a = 3; int b = 4; int c = a + b;
Integer的默認(rèn)值是null,int的默認(rèn)值是0
Integer變量使用之前必須先實(shí)例化,int變量可以直接使用。
Integer如果沒(méi)有實(shí)例化就直接使用會(huì)報(bào)空指針異常(因默認(rèn)值為null);int如果沒(méi)有初始化則使用默認(rèn)值0。
三、相同點(diǎn)
取值范圍相同
最小值:Integer.MIN_VALUE= -2147483648 (-2的31次方)
最大值:Integer.MAX_VALUE= 2147483647 (2的31次方-1)
原因:眾所周知,整型變量在內(nèi)存中的字節(jié)數(shù)是4個(gè)字節(jié),一個(gè)字節(jié)是8位,因此一個(gè)整型變量占據(jù)32位,且首位為0表示非負(fù)數(shù),1表示負(fù)數(shù)。
四、使用equals()和==的區(qū)別
對(duì)于每個(gè)java開(kāi)發(fā)人員來(lái)說(shuō),Integer的用法再熟悉不過(guò)了,但仍然會(huì)有些工作一兩年的小伙伴依然搞不清楚不同情況下equals()方法和 == 等號(hào)究竟有什么結(jié)果,以至于在基礎(chǔ)的面試過(guò)程中翻跟頭體驗(yàn)屢試不爽。這次我們一次性把所有可能以及結(jié)果出清楚。
equals()
此方法由Object類(lèi)定義,在Integer類(lèi)中又對(duì)其進(jìn)行了重寫(xiě),我們來(lái)看一下重寫(xiě)后的方法。
public boolean equals(Object obj) { if (obj instanceof Integer) { return value == ((Integer)obj).intValue(); } return false; }
這個(gè)方法先判斷參數(shù)對(duì)象是否繼承了Integer引用類(lèi)型(如果參數(shù)是int基本類(lèi)型,會(huì)先將其自動(dòng)裝箱成引用類(lèi)型),如果沒(méi)有繼承Integer引用類(lèi)型,則返false,這個(gè)很好理解,如果數(shù)字和字符相比,那叫什么事嘛。如果繼承了Integer引用類(lèi)型,就先把它強(qiáng)轉(zhuǎn)成Integer類(lèi)型,再通過(guò)intValue()方法將其拆箱為int基本類(lèi)型,再和value進(jìn)行地址比較(==右邊的value表示的是此Integer對(duì)象在常量池中的值,該值其實(shí)也是及本來(lái)想常量)。
? 說(shuō)了一大堆,簡(jiǎn)單一句話(huà)就是equals()方法進(jìn)行的是常量池中的地址是否相等。
? 另外一點(diǎn):常量池中值相等的數(shù)據(jù)不會(huì)同時(shí)出現(xiàn),常量池中只保存一份相同的值。
==
比較兩個(gè)對(duì)象在內(nèi)存中的地址是否相等。
下面是我們使用equals()方法和==方法來(lái)判斷Integer和int的區(qū)別。在此之前,我們要先知道裝箱和拆箱、常量池對(duì)象和堆內(nèi)存對(duì)象兩個(gè)概念:
裝箱
將基本數(shù)據(jù)類(lèi)型變成引用類(lèi)型的過(guò)程。發(fā)生在將基本數(shù)據(jù)類(lèi)型賦值給引用類(lèi)型的時(shí)候, 如Integer i = 5;
拆箱
將引用類(lèi)型變成基本數(shù)據(jù)類(lèi)型的過(guò)程。發(fā)生在將引用類(lèi)型和基本數(shù)據(jù)類(lèi)型做運(yùn)算的時(shí)候,如下所示
Integer i = 5; int j = 5; System.out.println(i==j);
常量池對(duì)象
保存的是創(chuàng)建對(duì)象時(shí)的字面量,相等的字面量只保存一次
堆內(nèi)存對(duì)象
保存的是new一個(gè)對(duì)象時(shí)這個(gè)對(duì)象在堆內(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)的過(guò)程中,java為我們做了兩件事,先在常量池中創(chuàng)建一個(gè)int=1的常量,再在堆內(nèi)存中創(chuàng)建一個(gè)對(duì)這個(gè)常量的地址引用,因此在用equals()方法時(shí)比較的是常量池中int=1這個(gè)常量,自己和自己比較的結(jié)果當(dāng)然為true;
而在==運(yùn)算時(shí),直接比較了堆內(nèi)存中的地址,于是比較的結(jié)果為false
情況2
在Integer j = 1的過(guò)程中,實(shí)際上是java為我們?cè)诔A砍刂猩闪艘粋€(gè)常量對(duì)象1,然后j對(duì)象指向這個(gè)常量對(duì)象的地址。
equals()運(yùn)算結(jié)果為true是因?yàn)楸容^的是常量池中對(duì)象的比較,實(shí)際上常量池中只有一個(gè)值為1的對(duì)象,即自己比較自己。
==運(yùn)算結(jié)果為false是因?yàn)楸容^的是堆內(nèi)存中對(duì)象j的地址和常量池中常量對(duì)象1的地址,是兩個(gè)不同的地址比較
情況3
equals()運(yùn)算結(jié)果為true可以參考上面的源碼以及下方解釋
==運(yùn)算結(jié)果為true是因?yàn)橐妙?lèi)型Integer和基本數(shù)據(jù)類(lèi)型int比較時(shí),引用類(lèi)型會(huì)自動(dòng)拆包裝為int,然后進(jìn)行比較,實(shí)際上就變?yōu)閮蓚€(gè)int變量的比較,也是堆內(nèi)存中常量自己與自己的地址比較
情況4
變量i與變量j指向的其實(shí)都是常量池中的常量對(duì)象1,因此equals()運(yùn)算和==運(yùn)算的結(jié)果都是true
情況5
與情況4相同, 如果有疑惑可在文章后面查找答案
情況6
基本數(shù)據(jù)類(lèi)型只能進(jìn)行加、減、乘、除、等于五種運(yùn)算,沒(méi)有equals()方法,==運(yùn)算為true還是因?yàn)槎褍?nèi)存中常量自己與自己進(jìn)行地址比較。
情況7
new Integer(128)會(huì)先在常量池中創(chuàng)建一個(gè)常量對(duì)象128,再在堆內(nèi)存中保存這個(gè)常量對(duì)象的地址,Integer i = new Integer(128)中變量i指向堆內(nèi)存中保存這個(gè)常量對(duì)象地址的地址,它是指向堆內(nèi)存的??梢岳斫鉃閮蓚€(gè)不同的堆內(nèi)存地址指向相同的常量池地址。
equals()運(yùn)算結(jié)果為true是因?yàn)楸容^的是常量池中的地址,而該地址只有一個(gè),自己比較自己,結(jié)果為true
==運(yùn)算比較的是這兩個(gè)堆內(nèi)存的地址,因此結(jié)果為false
情況8
equals()運(yùn)算結(jié)果為true說(shuō)了很多遍了,都是同一個(gè)原因
==運(yùn)算為false是因?yàn)楸容^的是堆內(nèi)存中兩個(gè)對(duì)象的地址,原因和情況5相同,可在文章后面查找答案
情況9
與情況3相同
情況10
與情況8相同,可在文章后面查找答案
情況11
與情況3相同
情況12
與情況6相同
五、解惑
JVM中一個(gè)字節(jié)一下的整型數(shù)據(jù)(即[128,127])會(huì)在JVM啟動(dòng)時(shí)加載進(jìn)內(nèi)存,除非用new Integer()顯示的創(chuàng)建對(duì)象,否則都是同一對(duì)象。當(dāng)使用Integer i = 1時(shí),會(huì)將1這個(gè)數(shù)字進(jìn)行緩存,下次再運(yùn)行Integer j = 1時(shí),就會(huì)直接從緩存中取,就不會(huì)new了, 這樣取到的1 地址也是相同的, ==判斷返回true。
對(duì)于-128 ~ 127范圍之外的數(shù) 實(shí)際上就是new得到的, 地址不同==判斷為false
java在編譯Integer i = 100 ;時(shí),java內(nèi)部會(huì)將其翻譯成為Integer i = Integer.valueOf(100);
所以關(guān)鍵就是看valueOf()函數(shù)了。JDK源碼的valueOf()是這樣的
到此,Integer與int運(yùn)算時(shí)的迷之操作總算解釋清楚了,如果有需要指正或者補(bǔ)充的,請(qǐng)小伙伴們掃下方二維碼關(guān)注公眾號(hào)并在后臺(tái)留言,我會(huì)及時(shí)給予回復(fù),讓我們一起把java學(xué)到荒,頭發(fā)掉到光。
到此這篇關(guān)于Java中Integer和int的區(qū)別解讀的文章就介紹到這了,更多相關(guān)Integer和int的區(qū)別內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java中關(guān)于int和Integer的區(qū)別詳解
- Java int與integer的對(duì)比區(qū)別
- Java中int和Integer的區(qū)別
- 詳解Java中int和Integer的區(qū)別
- Java中int與integer的區(qū)別(基本數(shù)據(jù)類(lèi)型與引用數(shù)據(jù)類(lèi)型)
- 通過(guò)實(shí)例了解Java Integer類(lèi)和int的區(qū)別
- Java數(shù)據(jù)類(lèi)型Integer與int的區(qū)別詳細(xì)解析
- 簡(jiǎn)單了解java中int和Integer的區(qū)別
- Java中int與Integer的區(qū)別詳解及常見(jiàn)問(wèn)題解決
相關(guān)文章
spring boot中使用RabbitMQ routing路由詳解
本篇文章主要介紹了spring boot中使用RabbitMQ routing路由詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03Java Map遍歷2種實(shí)現(xiàn)方法代碼實(shí)例
這篇文章主要介紹了Java Map遍歷2種實(shí)現(xiàn)方法代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10mybatis中返回多個(gè)map結(jié)果問(wèn)題
這篇文章主要介紹了mybatis中返回多個(gè)map結(jié)果問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06SpringBoot 增量部署發(fā)布的實(shí)現(xiàn)步驟
本文介紹了通過(guò)拆分項(xiàng)目jar包和使用類(lèi)加載器實(shí)現(xiàn)Spring Boot的增量部署,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-12-12MyBatis-Plus流式查詢(xún)的實(shí)現(xiàn)示例
MyBatis-Plus 從 3.5.4 版本開(kāi)始支持流式查詢(xún),通過(guò)ResultHandler接口實(shí)現(xiàn)結(jié)果集的流式查詢(xún),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-12-12spring boot多數(shù)據(jù)源動(dòng)態(tài)切換代碼實(shí)例
這篇文章主要介紹了spring boot多數(shù)據(jù)源動(dòng)態(tài)切換代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01???????Spring多租戶(hù)數(shù)據(jù)源管理 AbstractRoutingDataSource
本文技術(shù)了???????Spring多租戶(hù)數(shù)據(jù)源管理 AbstractRoutingDataSource,下文詳細(xì)內(nèi)容介紹,需要的小伙伴可以參考一下2022-05-05