亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java兩個(gè)integer數(shù)據(jù)判斷相等用==還是equals

 更新時(shí)間:2021年12月24日 11:31:42   作者:lm  
本文主要介紹了java兩個(gè)integer數(shù)據(jù)判斷相等用==還是equals,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

問(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)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 基于 SASL/SCRAM 讓 Kafka 實(shí)現(xiàn)動(dòng)態(tài)授權(quán)認(rè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-07
  • Java中Map集合中的Entry對(duì)象用法

    Java中Map集合中的Entry對(duì)象用法

    這篇文章主要介紹了Java中Map集合中的Entry對(duì)象用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-09-09
  • JavaSE實(shí)現(xiàn)文件壓縮與解壓縮的技巧分享

    JavaSE實(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-03
  • 深入了解Java I/O 之File類(lèi)

    深入了解Java I/O 之File類(lèi)

    這篇文章主要介紹了Java I/O深入學(xué)習(xí)之File和RandomAccessFile, I/O系統(tǒng)即輸入/輸出系統(tǒng),對(duì)于一門(mén)程序語(yǔ)言來(lái)說(shuō),創(chuàng)建一個(gè)好的輸入/輸出系統(tǒng)并非易事。需要的朋友可以參考下
    2021-08-08
  • Java中如何獲取mysql連接的3種方法總結(jié)

    Java中如何獲取mysql連接的3種方法總結(jié)

    最近工作中需要用到mysql連接,發(fā)現(xiàn)實(shí)現(xiàn)的方法不止一個(gè),所以就來(lái)總結(jié)下,下面這篇文章主要給大家介紹了關(guān)于Java中如何獲取mysql連接的3種方法,需要的朋友可以參考借鑒,感興趣的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-08-08
  • springmvc流程圖以及配置解析

    springmvc流程圖以及配置解析

    這篇文章主要介紹了springmvc流程圖以及配置解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-09-09
  • Java計(jì)時(shí)器工具StopWatch的具體使用

    Java計(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-04
  • Java中使用LocalDate根據(jù)日期來(lái)計(jì)算年齡的實(shí)現(xiàn)方法

    Java中使用LocalDate根據(jù)日期來(lái)計(jì)算年齡的實(shí)現(xiàn)方法

    這篇文章主要介紹了Java中使用LocalDate根據(jù)日期來(lái)計(jì)算年齡的實(shí)現(xiàn)方法,需要的朋友可以參考下
    2018-01-01
  • Netty中的心跳檢測(cè)機(jī)制詳解

    Netty中的心跳檢測(cè)機(jī)制詳解

    這篇文章主要介紹了Netty中的心跳檢測(cè)機(jī)制詳解,Netty 是 基于 TCP 協(xié)議開(kāi)發(fā)的,在四層協(xié)議 TCP 協(xié)議的實(shí)現(xiàn)中也提供了 keepalive 報(bào)文用來(lái)探測(cè)對(duì)端是否可用,TCP 層將在定時(shí)時(shí)間到后發(fā)送相應(yīng)的 KeepAlive 探針以確定連接可用性,需要的朋友可以參考下
    2023-12-12
  • 深入學(xué)習(xí)java ThreadLocal的源碼知識(shí)

    深入學(xué)習(xí)java ThreadLocal的源碼知識(shí)

    ThreadLocal是一個(gè)本地線程副本變量工具類(lèi)。主要用于將私有線程和該線程存放的副本對(duì)象做一個(gè)映射,各個(gè)線程之間的變量互不干擾,特別適用于各個(gè)線程依賴(lài)不通的變量值完成操作的場(chǎng)景。下面我們來(lái)詳細(xì)了解一下它吧
    2019-06-06

最新評(píng)論