java兩個(gè)integer數(shù)據(jù)判斷相等用==還是equals
問(wèn)題案例
來(lái)個(gè)簡(jiǎn)單點(diǎn)的例子
public static void main(String[] args) { for (int i = 0; i < 150; i++) { Integer a = i; Integer b = i; System.out.println(i + " " + (a == b)); } }
i取值從0到150,每次循環(huán)a與b的數(shù)值均相等,輸出a == b。運(yùn)行結(jié)果:
0 true
1 true
2 true
3 true
...
126 true
127 true
128 false
129 false
130 false
...
從128開(kāi)始a和b就不再相等了。
原因分析
首先回顧一下自動(dòng)裝箱。對(duì)于下面這行代碼
Integer a = 1;
變量a為Integer類(lèi)型,而1為int類(lèi)型,且Integer和int之間并無(wú)繼承關(guān)系,按照J(rèn)ava的一般處理方法,這行代碼應(yīng)該報(bào)錯(cuò)。
但因?yàn)樽詣?dòng)裝箱機(jī)制的存在,在為Integer類(lèi)型的變量賦int類(lèi)型值時(shí),Java會(huì)自動(dòng)將int類(lèi)型轉(zhuǎn)換為Integer類(lèi)型,即
Integer a = Integer.valueOf(1);
valueOf()方法返回一個(gè)Integer類(lèi)型值,并將其賦值給變量a。這就是int的自動(dòng)裝箱。
再看最開(kāi)始的例子:
public static void main(String[] args) { for (int i = 0; i < 150; i++) { Integer a = i; Integer b = i; System.out.println(i + " " + (a == b)); } }
每次循環(huán)時(shí),Integer a = i和Integer b = i都會(huì)觸發(fā)自動(dòng)裝箱,而自動(dòng)裝箱會(huì)將int轉(zhuǎn)換Integer類(lèi)型值并返回;我們知道Java中兩個(gè)new出來(lái)的對(duì)象因?yàn)闀r(shí)不同的實(shí)例,無(wú)論如何==都會(huì)返回fasle。比如
new Integer(1) == new Integer(1);
就會(huì)返回false。
那么例子中Integer a = i和Integer b = i自動(dòng)裝箱產(chǎn)生的變量a和b就不應(yīng)該時(shí)同一個(gè)對(duì)象了,那么==的結(jié)果應(yīng)該時(shí)false。128以上為false容易理解,但為何0到127時(shí)返回true了呢?==返回true的唯一情況是比較的兩個(gè)對(duì)象為同一個(gè)對(duì)象,那不妨把例子中a和b的內(nèi)存地址都打印出來(lái)看看:
for(int i=0;i<150;i++){ Integer a=i; Integer b=i; System.out.println(a+" "+b+" "+System.identityHashCode(a)+" "+System.identityHashCode(b)); }
identityHashCode()方法可以理解為輸出對(duì)應(yīng)變量的內(nèi)存地址,輸出為:
0 0 762119098 762119098
1 1 1278349992 1278349992
2 2 1801910956 1801910956
3 3 1468253089 1468253089
...
126 126 1605164995 1605164995
127 127 1318497351 1318497351
128 128 101224864 479240824
129 129 1373088356 636728630
130 130 587071409 1369296745
...
竟然從0到127不同時(shí)候自動(dòng)裝箱得到的是同一個(gè)對(duì)象!從128開(kāi)始才是正常情況。
源碼分析
“從0到127不同時(shí)候自動(dòng)裝箱得到的是同一個(gè)對(duì)象”就只能有一種解釋?zhuān)鹤詣?dòng)裝箱并不一定new出新的對(duì)象。
既然自動(dòng)裝箱涉及到的方法是Integer.valueOf(),不妨看看其源代碼:
public static Integer valueOf(int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); }
其注釋里就直接說(shuō)明了-128到127之間的值都是直接從緩存中取出的??纯词窃趺磳?shí)現(xiàn)的:如果int型參數(shù)i在IntegerCache.low和IntegerCache.high范圍內(nèi),則直接由IntegerCache返回;否則new一個(gè)新的對(duì)象返回。似乎IntegerCache.low就是-128,IntegerCache.high就是127了
IntegerCache的源碼:
private static class IntegerCache { static final int low = -128; static final int high; static final Integer cache[]; static { // high value may be configured by property int h = 127; String integerCacheHighPropValue = sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); if (integerCacheHighPropValue != null) { try { int i = parseInt(integerCacheHighPropValue); i = Math.max(i, 127); // Maximum array size is Integer.MAX_VALUE h = Math.min(i, Integer.MAX_VALUE - (-low) -1); } catch( NumberFormatException nfe) { // If the property cannot be parsed into an int, ignore it. } } high = h; cache = new Integer[(high - low) + 1]; int j = low; for(int k = 0; k < cache.length; k++) cache[k] = new Integer(j++); // range [-128, 127] must be interned (JLS7 5.1.7) assert IntegerCache.high >= 127; } private IntegerCache() {} }
果然在其static塊中就一次性生成了-128到127直接的Integer類(lèi)型變量存儲(chǔ)在cache[]中,對(duì)于-128到127之間的int類(lèi)型,返回的都是同一個(gè)Integer類(lèi)型對(duì)象。
這下真相大白了,整個(gè)工作過(guò)程就是:Integer.class在裝載(Java虛擬機(jī)啟動(dòng))時(shí),其內(nèi)部類(lèi)型IntegerCache的static塊即開(kāi)始執(zhí)行,實(shí)例化并暫存數(shù)值在-128到127之間的Integer類(lèi)型對(duì)象。當(dāng)自動(dòng)裝箱int型值在-128到127之間時(shí),即直接返回IntegerCache中暫存的Integer類(lèi)型對(duì)象
解決方法
既然我們的目的是比較數(shù)值是否相等,而非判斷是否為同一對(duì)象;而自動(dòng)裝箱又不能保證同一數(shù)值的Integer一定是同一對(duì)象或一定不是同一對(duì)象,那么就不要用==,直接用equals()好了。實(shí)際上,Integer重寫(xiě)了equals()方法,直接比較對(duì)象的數(shù)值是否相等。
for (int i = 0; i < 150; i++) { Integer a = i; Integer b = i; System.out.println(i + " " + (a.equals(b))); } //這樣返回值就全都是true了。 private final int value; public boolean equals(Object obj) { if (obj instanceof Integer) { return value == ((Integer)obj).intValue(); } return false; } public int intValue() { return value; }
備注
不僅int,Java中的另外7中基本類(lèi)型都可以自動(dòng)裝箱和自動(dòng)拆箱,其中也有用到緩存。見(jiàn)下表:
基本類(lèi)型 | 裝箱類(lèi)型 | 取值范圍 | 是否緩存 | 緩存范圍 |
---|---|---|---|---|
byte | Byte | -128 ~ 127 | 是 | -128 ~ 127 |
short | Short | -2^15 ~ (2^15 - 1) | 是 | -128 ~ 127 |
int | Integer | -2^31 ~ (2^31 - 1) | 是 | -128 ~ 127 |
long | Long | -2^63 ~ (2^63 - 1) | 是 | -128~127 |
float | Float | -- | 否 | -- |
double | Double | -- | 否 | -- |
boolean | Boolean | true, false | 是 | true, false |
char | Character | \u0000 ~ \uffff |
到此這篇關(guān)于java兩個(gè)integer數(shù)據(jù)判斷相等用==還是equals的文章就介紹到這了,更多相關(guān)java integer判斷相等內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java LinkedHashSet集合的底層原理和TreeSet集合
- Java中HashMap和HashSet的高效使用技巧分享
- Java中的Set接口實(shí)現(xiàn)類(lèi)HashSet和LinkedHashSet詳解
- Java集合ArrayList、LinkedList、HashMap、HashSet最大容量
- Java中HashSet、LinkedHashSet和TreeSet區(qū)別詳解
- java的==運(yùn)算符和equals操作詳解
- Java中==和equals()的區(qū)別總結(jié)
- 詳解Java中==和equals()的區(qū)別
- 淺談java字符串比較到底應(yīng)該用==還是equals
- java中的HashSet與 == 和 equals的區(qū)別示例解析
相關(guān)文章
基于 SASL/SCRAM 讓 Kafka 實(shí)現(xiàn)動(dòng)態(tài)授權(quán)認(rèn)證的方法
在大數(shù)據(jù)處理和分析中?Apache Kafka?已經(jīng)成為了一個(gè)核心組件,本文將從零開(kāi)始部署?ZooKeeper?和?Kafka?并通過(guò)配置?SASL/SCRAM?和?ACL(訪問(wèn)控制列表)來(lái)增強(qiáng)?Kafka?的安全性,需要的朋友可以參考下2024-07-07JavaSE實(shí)現(xiàn)文件壓縮與解壓縮的技巧分享
我們?cè)谌粘?shí)際開(kāi)發(fā)中,對(duì)于文件壓縮和解壓縮場(chǎng)景,是非常常見(jiàn)的操作,本文詳細(xì)介紹JavaSE中文件壓縮和解壓縮的實(shí)現(xiàn)方法,包括源代碼解析、應(yīng)用場(chǎng)景案例、優(yōu)缺點(diǎn)分析、案例演示、文末總結(jié)等等,請(qǐng)同學(xué)們耐心閱讀2024-03-03Java計(jì)時(shí)器工具StopWatch的具體使用
計(jì)時(shí)器在很多地方都可以用到,本文主要介紹了Java計(jì)時(shí)器工具StopWatch的具體使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04Java中使用LocalDate根據(jù)日期來(lái)計(jì)算年齡的實(shí)現(xiàn)方法
這篇文章主要介紹了Java中使用LocalDate根據(jù)日期來(lái)計(jì)算年齡的實(shí)現(xiàn)方法,需要的朋友可以參考下2018-01-01深入學(xué)習(xí)java ThreadLocal的源碼知識(shí)
ThreadLocal是一個(gè)本地線程副本變量工具類(lèi)。主要用于將私有線程和該線程存放的副本對(duì)象做一個(gè)映射,各個(gè)線程之間的變量互不干擾,特別適用于各個(gè)線程依賴(lài)不通的變量值完成操作的場(chǎng)景。下面我們來(lái)詳細(xì)了解一下它吧2019-06-06