欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

為什么說HashMap線程不安全

 更新時間:2023年04月27日 08:58:49   作者:Cosolar  
本文主要介紹了為什么說HashMap線程不安全,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

在Java中,HashMap是一種常用的數(shù)據(jù)結(jié)構,它以鍵值對的形式存儲和管理數(shù)據(jù)。然而,由于HashMap在多線程環(huán)境下存在線程安全問題,因此在使用時需要格外小心。

簡單來說:在 hashMap1.7 中擴容的時候,因為采用的是頭插法,所以會可能會有循環(huán)鏈表產(chǎn)生,導致數(shù)據(jù)有問題,在 1.8 版本已修復,改為了尾插法; 在任意版本的 hashMap 中,如果在插入數(shù)據(jù)時多個線程命中了同一個槽,可能會有數(shù)據(jù)覆蓋的情況發(fā)生,導致線程不安全。

HashMap的線程不安全主要體現(xiàn)在以下兩個方面:

1. 并發(fā)修改導致數(shù)據(jù)不一致

HashMap的數(shù)據(jù)結(jié)構是基于數(shù)組和鏈表實現(xiàn)的。在進行插入或刪除操作時,如果不同線程同時修改同一個位置的元素,就會導致數(shù)據(jù)不一致的情況。具體來說,當兩個線程同時進行插入操作時,假設它們都要插入到同一個數(shù)組位置,并且該位置沒有元素,那么它們都會認為該位置可以插入元素,最終就會導致其中一個線程的元素被覆蓋掉。此外,在進行刪除操作時,如果兩個線程同時刪除同一個元素,也會導致數(shù)據(jù)不一致的情況。

以下是一個示例代碼,展現(xiàn)了兩個線程對HashMap進行并發(fā)修改的情況:

import java.util.HashMap;
public class HashMapThreadUnsafeExample {
    public static void main(String[] args) throws InterruptedException {
        final HashMap<String, Integer> map = new HashMap<>();
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                map.put("key" + i, i);
            }
        });
        Thread t2 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                map.put("key" + i, i * 2);
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("map size: " + map.size());
    }
}

上述示例代碼中,t1線程和t2線程都向HashMap中插入數(shù)據(jù),由于它們在進行插入操作時修改的是同一個位置的元素,因此最終導致了部分數(shù)據(jù)不一致的情況。例如,當t1線程插入了(key1, 1)以后,t2線程又插入了(key1, 2),這就導致了(key1, 1)被覆蓋掉,最終HashMap的大小只有10000而不是20000。

2. 并發(fā)擴容導致死循環(huán)或數(shù)據(jù)丟失

當HashMap的元素數(shù)量達到一定閾值時,它會觸發(fā)擴容操作,即重新分配更大的數(shù)組并將原來的元素重新映射到新的數(shù)組上。然而,在進行擴容操作時,如果不加鎖或者加鎖不正確,就可能導致死循環(huán)或者數(shù)據(jù)丟失的情況。具體來說,當兩個線程同時進行擴容操作時,它們可能會同時將某個元素映射到新的數(shù)組上,從而導致該元素被覆蓋掉。此外,在進行擴容操作時,如果線程不安全地修改了next指針,就可能會導致死循環(huán)的情況。

以下是一個示例代碼,展現(xiàn)了兩個線程對HashMap進行并發(fā)擴容的情況:

import java.util.HashMap;
public class HashMapThreadUnsafeExample {
    public static void main(String[] args) throws InterruptedException {
        final HashMap<String, Integer> map = new HashMap<>(2, 0.75f);
        map.put("key1", 1);
        map.put("key2", 2);
        map.put("key3", 3);
        Thread t1 = new Thread(() -> {
            for (int i = 4; i < 10000; i++) {
                map.put("key" + i, i);
            }
        });
        Thread t2 = new Thread(() -> {
            for (int i = 4; i < 10000; i++) {
                map.put("key" + i, i * 2);
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("map size: " + map.size());
    }
}

上述示例代碼中,t1線程和t2線程都向HashMap中插入數(shù)據(jù),并且HashMap被初始化為大小為2,負載因子為0.75,這就意味著HashMap在元素數(shù)量達到3時就會進行擴容操作。由于t1和t2線程同時進行擴容操作,它們有可能都將某個元素映射到新的數(shù)組上,導致該元素被覆蓋掉。此外,在進行擴容操作時,如果線程不安全地修改了next指針,就可能會導致死循環(huán)的情況。

除了并發(fā)修改和并發(fā)擴容外,還有以下情況可能導致HashMap不安全:

3. 非線程安全的迭代器

當使用非線程安全的迭代器遍歷HashMap時,如果在遍歷的過程中其他線程修改了HashMap的結(jié)構,就可能拋出ConcurrentModificationException異常。

以下是一個示例代碼,展現(xiàn)了如何通過多線程遍歷HashMap以及導致線程不安全的情況:

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
public class HashMapThreadUnsafeExample {
    public static void main(String[] args) throws InterruptedException {
        final Map<String, Integer> map = new HashMap<>();
        for (int i = 0; i < 10000; i++) {
            map.put("key" + i, i);
        }
        Thread t1 = new Thread(() -> {
            Iterator<Map.Entry<String, Integer>> iterator = map.entrySet().iterator();
            while (iterator.hasNext()) {
                System.out.println(iterator.next().getValue());
            }
        });
        Thread t2 = new Thread(() -> {
            for (int i = 10000; i < 20000; i++) {
                map.put("key" + i, i);
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
    }
}

上述示例代碼中,t1線程遍歷了HashMap中的元素,但并沒有對其進行加鎖保護。同時,在t1線程遍歷的過程中,t2線程又進行了另外一部分元素的插入操作,這就導致了HashMap結(jié)構的不穩(wěn)定性,最終可能會拋出ConcurrentModificationException異常。

4. 非線程安全的比較器

當使用非線程安全的比較器來定義HashMap的排序規(guī)則時,就可能導致在并發(fā)環(huán)境下出現(xiàn)數(shù)據(jù)不一致性的情況。

以下是一個示例代碼,展現(xiàn)了如何通過多線程修改HashMap中元素順序以及導致線程不安全的情況:

import java.util.Comparator;
import java.util.HashMap;
import java.util.Map;
public class HashMapThreadUnsafeExample {
    public static void main(String[] args) throws InterruptedException {
        final Map<String, Integer> map = new HashMap<>();
        map.put("key1", 1);
        map.put("key2", 2);
        map.put("key3", 3);
        Comparator<String> comparator = (s1, s2) -> {
            int i1 = Integer.parseInt(s1.substring(3));
            int i2 = Integer.parseInt(s2.substring(3));
            return Integer.compare(i1, i2);
        };
        Thread t1 = new Thread(() -> {
            for (int i = 4; i < 10000; i++) {
                map.put("key" + i, i);
            }
        });
        Thread t2 = new Thread(() -> {
            for (int i = 4; i < 10000; i++) {
                map.put("key" + i, i * 2);
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("map: " + map);
    }
}

上述示例代碼中,HashMap的排序規(guī)則使用了一個基于字符串處理的比較器來定義。當t1線程和t2線程同時進行插入操作時,由于它們在不同的元素上執(zhí)行修改操作,因此并不會出現(xiàn)ConcurrentModificationException異常。然而,由于比較器不是線程安全的,當t1和t2線程同時進行對相同的元素值進行賦值操作時,就可能導致HashMap結(jié)構的不穩(wěn)定性。例如,當t1線程將"key5"的值修改為5時,t2線程可能只修改到"value"字段的一部分,因此最終HashMap中的值可能出現(xiàn)混亂的情況。

寫到這里我想告訴大家:HashMap在多線程環(huán)境下存在線程安全問題,具體表現(xiàn)為并發(fā)修改導致數(shù)據(jù)不一致和并發(fā)擴容導致死循環(huán)或數(shù)據(jù)丟失。因此,在使用HashMap時需要采取相應的線程安全措施,例如使用ConcurrentHashMap、加鎖等。

到此這篇關于為什么說HashMap線程不安全的文章就介紹到這了,更多相關HashMap線程不安全內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • JAVA日期處理類詳解

    JAVA日期處理類詳解

    這篇文章主要介紹了Java實現(xiàn)的日期處理類,結(jié)合完整實例形式分析了Java針對日期的獲取、運算、轉(zhuǎn)換等相關操作技巧,需要的朋友可以參考下
    2021-08-08
  • java模擬hibernate一級緩存示例分享

    java模擬hibernate一級緩存示例分享

    這篇文章主要介紹了java模擬hibernate一級緩存示例,需要的朋友可以參考下
    2014-03-03
  • 詳解Spring中Bean的加載的方法

    詳解Spring中Bean的加載的方法

    本篇文章主要介紹了Spring中Bean的加載的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • 記錄一次connection reset 錯誤的解決全過程

    記錄一次connection reset 錯誤的解決全過程

    這篇文章主要介紹了記錄一次connection reset 錯誤的解決全過程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • java獲取系統(tǒng)路徑字體、得到某個目錄下的所有文件名、獲取當前路徑

    java獲取系統(tǒng)路徑字體、得到某個目錄下的所有文件名、獲取當前路徑

    這篇文章主要介紹了java獲取系統(tǒng)路徑字體、得到某個目錄下的所有文件名、獲取當前路徑,需要的朋友可以參考下
    2014-04-04
  • 關于Java中XML Namespace 命名空間問題

    關于Java中XML Namespace 命名空間問題

    這篇文章主要介紹了Java中XML Namespace 命名空間,XML命名空間是由國際化資源標識符 (IRI) 標識的 XML 元素和屬性集合,該集合通常稱作 XML“詞匯”,對XML Namespace 命名空間相關知識感興趣的朋友一起看看吧
    2021-08-08
  • Java如何自定義類數(shù)組的創(chuàng)建和初始化

    Java如何自定義類數(shù)組的創(chuàng)建和初始化

    這篇文章主要介紹了Java如何自定義類數(shù)組的創(chuàng)建和初始化,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Springboot shiro認證授權實現(xiàn)原理及實例

    Springboot shiro認證授權實現(xiàn)原理及實例

    這篇文章主要介紹了Springboot shiro認證授權實現(xiàn)原理及實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-06-06
  • 利用AOP實現(xiàn)系統(tǒng)告警的方法詳解

    利用AOP實現(xiàn)系統(tǒng)告警的方法詳解

    在開發(fā)的過程中會遇到各種各樣的開發(fā)問題,服務器宕機、網(wǎng)絡抖動、代碼本身的bug等等。針對代碼的bug,我們可以提前預支,通過發(fā)送告警信息來警示我們?nèi)ジ深A,盡早處理。本文將利用AOP實現(xiàn)系統(tǒng)告警,需要的可以參考一下
    2022-09-09
  • SpringMVC對日期類型的轉(zhuǎn)換示例

    SpringMVC對日期類型的轉(zhuǎn)換示例

    本篇文章主要介紹了SpringMVC對日期類型的轉(zhuǎn)換示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-02-02

最新評論