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

Java中使用HashMap時(shí)指定初始化容量性能解析

 更新時(shí)間:2023年02月05日 10:50:55   作者:xindoo  
這篇文章主要為大家介紹了Java中使用HashMap時(shí)指定初始化容量性能解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

正文

一些Java編程老手在做CodeReview時(shí),都會(huì)告訴其他人,使用HashMap時(shí)建議指定容量大小,原因是指定容量后,代碼性能會(huì)更好一些。后來(lái)隨著阿里Java開發(fā)手冊(cè)在業(yè)內(nèi)廣為傳播,這一點(diǎn)早已深入人心,我自己也早已習(xí)慣在使用HashMap時(shí)指定容量大小。

但我今天突發(fā)奇想,想知道指定容量和不指定容量時(shí)性能究竟有多少的差異,測(cè)試部分測(cè)試數(shù)據(jù)的結(jié)果讓我大跌眼睛,有些情況下指定容量的性能還比不指定容量時(shí)差??! ,但其他部分還是很符合我之前的認(rèn)知的。

openjdk17和jmh單線程測(cè)試

先說(shuō)下我的測(cè)試平臺(tái)和測(cè)試方法,我使用了openjdk17和jmh單線程測(cè)試,測(cè)試代碼如下:

    @Benchmark
    @BenchmarkMode(Mode.Throughput)
    @Measurement(iterations = 2, time = 5)
    @Threads(1)
    @Fork(0)
    @Warmup(iterations = 1, time = 5)
    public void withoutCap() {
        Map<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < CAP; i++) {
            map.put(random.nextInt(), 1);
        }
    }
    @Benchmark
    @BenchmarkMode(Mode.Throughput)
    @Measurement(iterations = 2, time = 5)
    @Threads(1)
    @Fork(0)
    @Warmup(iterations = 1, time = 5)
    public void withCap() {
        Map<Integer, Integer> map = new HashMap<>(CAP);
        for (int i = 0; i < CAP; i++) {
            map.put(random.nextInt(), 1);
        }
    }

這里為了避免Java中小數(shù)據(jù)緩存,我特意使用了隨機(jī)數(shù)作為KEY,而VALUE一視同仁都使用了1。兩個(gè)方法就是新建一個(gè)HashMap并不斷往map里put數(shù)據(jù),唯一差異就是一個(gè)指定了CAP參數(shù)。 在我設(shè)置了不同參數(shù)后,得到了以下數(shù)據(jù)(越高越好):

數(shù)據(jù)量不指定容量(ops/s)指定容量(ops/s)
25109543324000032
42516175611813275
8107671765900641
1629783742987958
3212316371545394
64567643764260
256129350185540
10242747535799
10252719568466
409666819937
327688071177
65536377567

可以看出,容量16是個(gè)分水嶺,當(dāng)容量為16時(shí),二者幾乎沒(méi)啥差異,這也很容易理解,當(dāng)不指定容量時(shí)默認(rèn)初始容量就是16。

  • 但容量大于16時(shí),指定容量時(shí)的性能會(huì)高于不指定時(shí)的性能,隨著數(shù)量的增加,前者會(huì)比后者性能高出50%。
  • 但當(dāng)數(shù)據(jù)量小于16時(shí),不指定容量大小反而性能更高,最多甚至相差2倍,這就和我們之前的認(rèn)知不一樣了。

上面數(shù)據(jù)中還有個(gè)很奇怪的點(diǎn),那就是當(dāng)數(shù)據(jù)量為1025時(shí),性能居然還高于1024,而且差異巨大。就好比別人比你多干了1份活,但用的時(shí)間比你少一半。我跑了多次都是這個(gè)結(jié)果,這不是測(cè)試誤差,這個(gè)結(jié)果和計(jì)算機(jī)底層存儲(chǔ)實(shí)現(xiàn)有關(guān),具體原理可以參考問(wèn)題 為什么轉(zhuǎn)置512x512的矩陣比轉(zhuǎn)置513x513的矩陣慢?

備注:以上數(shù)據(jù)經(jīng)過(guò)多次運(yùn)行測(cè)試,數(shù)據(jù)雖有波動(dòng),但數(shù)據(jù)波動(dòng)基本都在3%以內(nèi)。

那為什么在大數(shù)據(jù)量的情況下,指定容量的代碼性能會(huì)更好呢?這就得說(shuō)到HashMap的實(shí)現(xiàn)原理,更詳細(xì)內(nèi)容可以參考我之前寫的HashMap源碼淺析。這里為了方便大家直觀地理解性能差異產(chǎn)生的原因,我們用牧場(chǎng)養(yǎng)羊類比下。

假設(shè)你要開始養(yǎng)羊,你得現(xiàn)有場(chǎng)地吧,假設(shè)你先找了塊小場(chǎng)地,但隨著你的羊群發(fā)展壯大,場(chǎng)地不夠用了,你就得搬到一個(gè)更大的新場(chǎng)地,如果發(fā)展速度特別快,你就得頻繁搬家,搬家就逐漸變成了負(fù)擔(dān)。但如果你一開始就知道你最多能養(yǎng)多少的羊,直接找個(gè)足夠大的場(chǎng)地,不就能省去一直搬家的成本了嗎!

這里你把羊類比成數(shù)據(jù),場(chǎng)地類比為內(nèi)存,在HashMap中,如果開始不指定容量大小,JVM默認(rèn)會(huì)給你一個(gè)非常小的(16)的容量空間,如果之后數(shù)據(jù)量變多,就需要重新申請(qǐng)更大的空間,并把數(shù)據(jù)遷移到新空間上,于是額外增加了時(shí)間消耗。這便是性能差異產(chǎn)生的原因。

但當(dāng)容量小于16時(shí),指定容量的方式反而性能更差。這個(gè)我之前從未看過(guò)其他資料有說(shuō)過(guò),我簡(jiǎn)單談下自己的分析和理解。 當(dāng)調(diào)用new HashMap()和new HashMap(CAP)時(shí),分別執(zhí)行了不同的構(gòu)造函數(shù),而二者的構(gòu)造函數(shù)的邏輯是有差異的,當(dāng)指定容量時(shí),執(zhí)行了容量參數(shù)檢查的代碼:

    public HashMap(int initialCapacity, float loadFactor) {
        if (initialCapacity < 0)
            throw new IllegalArgumentException("Illegal initial capacity: " +
                                               initialCapacity);
        if (initialCapacity > MAXIMUM_CAPACITY)
            initialCapacity = MAXIMUM_CAPACITY;
        if (loadFactor <= 0 || Float.isNaN(loadFactor))
            throw new IllegalArgumentException("Illegal load factor: " +
                                               loadFactor);
        this.loadFactor = loadFactor;
        this.threshold = tableSizeFor(initialCapacity);
    }
    static final int tableSizeFor(int cap) {
        int n = -1 >>> Integer.numberOfLeadingZeros(cap - 1);
        return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
    }

不指定容量時(shí),構(gòu)造方法內(nèi)只有一行this.loadFactor = DEFAULT_LOAD_FACTOR;,在put的數(shù)據(jù)量一致時(shí),后續(xù)所有的代碼執(zhí)行流程都是一致的,所以指定容量時(shí),上面容量參數(shù)檢查的代碼帶來(lái)了額外的性能負(fù)擔(dān),所以導(dǎo)致數(shù)據(jù)量較小時(shí)指定容量時(shí)反而性能更差一些。

總結(jié)

最后回到文章標(biāo)題上來(lái),Java中使用HashMap時(shí)指定初始化容量性能一定會(huì)更好嘛?答案是不一定,指定容量也有可能性能會(huì)更差。當(dāng)然,絕大多數(shù)情況下還是建議指定容量的,類似的還有ArrayList,也建議指定容量。 別人給出的結(jié)論不一定的完全正確的,只有知道產(chǎn)生結(jié)論的原因,才能更有效的利用這個(gè)結(jié)論。

以上就是Java中使用HashMap時(shí)指定初始化容量性能解析的詳細(xì)內(nèi)容,更多關(guān)于Java HashMap容量性能的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot整合Quartz方法詳解

    SpringBoot整合Quartz方法詳解

    這篇文章詳解介紹了SpringBoot整合Quartz的方法,Quartz是一個(gè)比較成熟了的定時(shí)任務(wù)框架,本文實(shí)例代碼給大家詳細(xì)講解,需要的朋友可以參考下
    2023-04-04
  • 使用Nacos實(shí)現(xiàn)動(dòng)態(tài)路由的步驟和代碼示例

    使用Nacos實(shí)現(xiàn)動(dòng)態(tài)路由的步驟和代碼示例

    這篇文章主要介紹了使用 Nacos 實(shí)現(xiàn) Spring Cloud Gateway 的動(dòng)態(tài)路由,本文給大家介紹了具體的實(shí)現(xiàn)步驟和代碼案例,感興趣的小伙伴跟著小編一起來(lái)看看吧
    2024-09-09
  • springboot多文件上傳代碼實(shí)例及解析

    springboot多文件上傳代碼實(shí)例及解析

    這篇文章主要介紹了springboot多文件上傳代碼實(shí)例及解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • 搭建Spring MVC和Vue3的應(yīng)用程序的實(shí)現(xiàn)

    搭建Spring MVC和Vue3的應(yīng)用程序的實(shí)現(xiàn)

    本文主要介紹了搭建Spring MVC和Vue3的應(yīng)用程序的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-11-11
  • 如何使用try-with-resource機(jī)制關(guān)閉連接

    如何使用try-with-resource機(jī)制關(guān)閉連接

    這篇文章主要介紹了使用try-with-resource機(jī)制關(guān)閉連接的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • springboot+mybatis配置clickhouse實(shí)現(xiàn)插入查詢功能

    springboot+mybatis配置clickhouse實(shí)現(xiàn)插入查詢功能

    這篇文章主要介紹了springboot+mybatis配置clickhouse實(shí)現(xiàn)插入查詢功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • 利用Spring?Boot和JPA創(chuàng)建GraphQL?API

    利用Spring?Boot和JPA創(chuàng)建GraphQL?API

    這篇文章主要介紹了利用Spring?Boot和JPA創(chuàng)建GraphQL?API,GraphQL既是API查詢語(yǔ)言,也是使用當(dāng)前數(shù)據(jù)執(zhí)行這些查詢的運(yùn)行時(shí),下文更多相關(guān)內(nèi)容介紹需要的小伙伴可以參考一下
    2022-04-04
  • springboot使用redis對(duì)單個(gè)對(duì)象進(jìn)行自動(dòng)緩存更新刪除的實(shí)現(xiàn)

    springboot使用redis對(duì)單個(gè)對(duì)象進(jìn)行自動(dòng)緩存更新刪除的實(shí)現(xiàn)

    本文主要介紹了springboot使用redis對(duì)單個(gè)對(duì)象進(jìn)行自動(dòng)緩存更新刪除的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Spring MVC框架配置方法詳解

    Spring MVC框架配置方法詳解

    這篇文章主要為大家詳細(xì)介紹了Spring MVC框架的配置方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-04-04
  • Java基礎(chǔ)之java處理ip的工具類

    Java基礎(chǔ)之java處理ip的工具類

    這篇文章主要介紹了Java基礎(chǔ)應(yīng)用,使用java處理ip的工具類的相關(guān)資料,需要的朋友可以參考下
    2014-10-10

最新評(píng)論