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

源碼分析ConcurrentHashMap如何保證線程安全

 更新時間:2023年06月28日 10:33:51   作者:小威要向諸佬學習呀  
這篇文章將結合底層源碼為大家詳細介紹一下ConcurrentHashMap是如何保證線程安全的,文中是示例代碼講解詳細,感興趣的小伙伴可以了解一下

JDK1.7保證線程安全

ConcurrentHashMap在JDK 1.7和JDK 1.8版本保證線程安全及其底層數(shù)據(jù)結構是不一樣的,這一塊是面試中的重點,接下來詳細介紹一下它們。

在JDK 1.7中,ConcurrentHashMap采用了分段鎖(Segment)的設計來保證線程安全。下面我們將通過詳細解讀其底層源碼,來介紹其線程安全實現(xiàn)原理。

ConcurrentHashMap的主要類是Segment。每個Segment是一個獨立的鎖,并且維護著一個HashEntry數(shù)組。HashEntry是鏈表節(jié)點,存儲了鍵值對。

首先,我們來看一下ConcurrentHashMap的基本數(shù)據(jù)結構:

static final class HashEntry<K, V> {
    final int hash;
    final K key;
    volatile V value;
    volatile HashEntry<K, V> next;
    HashEntry(int hash, K key, V value, HashEntry<K, V> next) {
        this.hash = hash;
        this.key = key;
        this.value = value;
        this.next = next;
    }
}
static final class Segment<K, V> extends ReentrantLock implements Serializable {
    static final float LOAD_FACTOR = 0.75f;
    transient volatile HashEntry<K, V>[] table;
    transient int count;
    transient int modCount;
    transient int threshold;
    final float loadFactor;
}

每個Segment都是一個繼承自ReentrantLock的可重入鎖,具備獨立的線程安全性。table是Segment內部的HashEntry數(shù)組,用于存儲鍵值對。count表示當前Segment中的元素數(shù)量,modCount用于記錄修改次數(shù),threshold表示擴容的閾值,loadFactor表示加載因子。

接下來,我們看一下ConcurrentHashMap的put操作:

public V put(K key, V value) {
    if (value == null)
        throw new NullPointerException();
    int hash = hash(key.hashCode());
    int segmentIndex = getSegmentIndex(hash);
    return segments[segmentIndex].put(key, hash, value, false);
}
final V put(K key, int hash, V value, boolean onlyIfAbsent) {
    lock(); // 獲取當前Segment的鎖
    try {
        int c = count;
        if (c++ > threshold) // 判斷是否需要擴容
            rehash();
        HashEntry<K, V>[] tab = table;
        int index = hash & (tab.length - 1);
        HashEntry<K, V> first = tab[index];
        HashEntry<K, V> e = first;
        while (e != null && (e.hash != hash || !key.equals(e.key))) 
            e = e.next;
        V oldValue;
        if (e != null) { // 鍵存在,更新值
            oldValue = e.value;
            if (!onlyIfAbsent)
                e.value = value;
        } else { // 鍵不存在,創(chuàng)建新節(jié)點并添加到鏈表頭部
            oldValue = null;
            ++modCount;
            tab[index] = new HashEntry<K, V>(hash, key, value, first);
            count = c; // 更新元素數(shù)量
        }
        return oldValue;
    } finally {
        unlock(); // 釋放當前Segment的鎖
    }
}

在put操作中,首先通過hash函數(shù)計算鍵的散列值hash,然后根據(jù)散列值獲取對應的Segment。接著,通過Segment的鎖保證了當前操作的線程安全。

在獲取到Segment的鎖之后,首先判斷當前Segment中的元素數(shù)量count是否超過了閾值threshold,如果超過了則進行擴容。然后通過散列值和數(shù)組長度計算出鍵對應的索引位置index,并從對應的鏈表開始遍歷,尋找是否存在相同的鍵。

如果找到了相同的鍵,則更新對應的值;如果沒有找到相同的鍵,則創(chuàng)建一個新的HashEntry節(jié)點,并將其添加到鏈表的頭部。

在完成操作后,釋放Segment的鎖。

通過分段鎖的設計,JDK 1.7的ConcurrentHashMap允許多個線程同時操作不同的Segment,從而提高了并發(fā)性能。雖然在高并發(fā)情況下仍可能存在競爭問題,但通過細粒度的鎖設計,可以減少鎖競爭的概率,提升整體性能。

JDK1.8保證線程安全

在JDK 1.8中,ConcurrentHashMap進行了重大改進,采用了更加高效的并發(fā)控制機制來保證線程安全。相較于JDK 1.7的分段鎖設計,JDK 1.8引入了基于CAS(Compare and Swap)操作和鏈表/紅黑樹結構的鎖機制以及其他優(yōu)化,大大提高了并發(fā)性能。

底層數(shù)據(jù)結構:

JDK 1.8中的ConcurrentHashMap采用了數(shù)組+鏈表/紅黑樹的結構。具體來說,它將整個哈希桶(Hash Bucket)劃分為若干個節(jié)點(Node)。每個節(jié)點代表一個存儲鍵值對的單元,可以是鏈表節(jié)點(普通節(jié)點)或紅黑樹節(jié)點(樹節(jié)點),這取決于節(jié)點內的鍵值對數(shù)量是否達到閾值。使用紅黑樹結構可以提高查找、插入、刪除等操作的效率。

主要類和數(shù)據(jù)結構如下:

static final class Node<K, V> implements Map.Entry<K, V> {
    final int hash;
    final K key;
    volatile V value;
    volatile Node<K, V> next;
    Node(int hash, K key, V value, Node<K, V> next) {
        this.hash = hash;
        this.key = key;
        this.value = value;
        this.next = next;
    }
}
static final class TreeNode<K, V> extends Node<K, V> {
    TreeNode(int hash, K key, V value, Node<K, V> next) {
        super(hash, key, value, next);
    }
    // 省略了紅黑樹相關的操作代碼
}
static final class ConcurrentHashMap<K, V> {
    transient volatile Node<K, V>[] table;
    transient volatile int sizeCtl;
    transient volatile int baseCount;
    transient volatile int modCount;
}

ConcurrentHashMap的線程安全實現(xiàn)原理:

初始狀態(tài):在初始狀態(tài)下,table為null,sizeCtl為0。當?shù)谝粋€元素被插入時,會根據(jù)并發(fā)級別(Concurrency Level)計算出數(shù)組的長度,并使用CAS操作將數(shù)組初始化為對應長度的桶。

插入操作

put方法:當進行插入操作時,ConcurrentHashMap首先計算鍵的散列值,然后根據(jù)散列值和數(shù)組長度計算出對應的桶位置。接著使用CAS操作嘗試插入新節(jié)點,如果成功則插入完成;如果失敗,則進入下一步。

resize方法:插入節(jié)點時,若發(fā)現(xiàn)鏈表中的節(jié)點數(shù)量已經(jīng)達到閾值(默認為8),則將鏈表轉化為紅黑樹,提高查找、插入、刪除等操作的效率。在轉化過程中,利用synchronized鎖住鏈表或紅黑樹所在的桶,并進行相應的操作。

forwardTable方法:若節(jié)點數(shù)量超過閾值(默認為64)且table未被初始化,則使用CAS操作將table指向擴容后的桶數(shù)組,并根據(jù)需要將鏈表或紅黑樹進行分割,以減小線程之間的沖突。

查詢操作

get方法:當進行查詢操作時,首先計算鍵的散列值,然后根據(jù)散列值和數(shù)組長度計算出對應的桶位置。接著從桶位置的鏈表或紅黑樹中查找對應的節(jié)點。

其他操作

remove方法:當進行刪除操作時,首先計算鍵的散列值,然后根據(jù)散列值和數(shù)組長度計算出對應的桶位置。接著使用synchronized鎖住桶,并進行相應的操作。

綜上所述,JDK 1.8的ConcurrentHashMap通過CAS操作、鎖機制(synchronized)以及鏈表/紅黑樹結構來保證線程安全。CAS操作用于插入新節(jié)點和初始化桶數(shù)組,鎖機制用于鏈表/紅黑樹的轉化和刪除操作,鏈表/紅黑樹結構用于提高查找、插入、刪除操作的效率。這些優(yōu)化措施使得ConcurrentHashMap在高并發(fā)環(huán)境下具有較好的性能表現(xiàn)。

JDK1.7和JDK1.8對比總結

在JDK 1.7和JDK 1.8中,ConcurrentHashMap有以下主要區(qū)別:

JDK 1.7中的實現(xiàn)方式:

  • JDK 1.7中的ConcurrentHashMap使用分段鎖(Segment Locking)的設計。它將整個哈希表分成多個段(Segment),每個段都有自己的鎖。這樣可以降低并發(fā)操作時鎖的爭用范圍,提高并發(fā)性能。
  • 每個段中包含一個HashEntry數(shù)組,每個HashEntry是一個鏈表結構,用于解決哈希沖突。
  • 由于每個段都有自己的鎖,不同的線程可以同時訪問不同的段,從而提高了并發(fā)度。

JDK 1.8中的改進:

JDK 1.8中的ConcurrentHashMap采用了CAS操作、鎖機制以及鏈表/紅黑樹結構的改進。

  • 數(shù)據(jù)結構改進:JDK 1.8中使用數(shù)組+鏈表/紅黑樹的結構,代替了JDK 1.7中的段+鏈表結構。數(shù)組用于存儲桶,鏈表/紅黑樹用于解決哈希沖突。
  • CAS操作:JDK 1.8使用CAS(Compare and Swap)操作來插入新節(jié)點和初始化桶數(shù)組。CAS操作是一種樂觀鎖機制,通過原子操作比較并交換的方式進行,并發(fā)安全性更好。
  • 鎖的改進:JDK 1.8中引入了基于CAS操作和鏈表/紅黑樹結構的鎖機制。對于鏈表/紅黑樹上的操作,使用synchronized鎖住桶,以保證操作的原子性。
  • 鏈表轉化為紅黑樹:JDK 1.8在插入操作時,當鏈表中的節(jié)點數(shù)量達到一定閾值時,會將鏈表轉化為紅黑樹,提高查找、插入、刪除等操作的效率。
  • resize操作的改進:JDK 1.8中的resize操作(擴容)采用了分割鏈表/紅黑樹的方式,減小了線程沖突的概率。

總的來說,JDK 1.8中的ConcurrentHashMap在數(shù)據(jù)結構、CAS操作、鎖機制和鏈表/紅黑樹結構等方面進行了改進,相較于JDK 1.7,性能更好且并發(fā)度更高。這些改進使得JDK 1.8中的ConcurrentHashMap在高并發(fā)環(huán)境下表現(xiàn)更優(yōu)秀。

到此這篇關于源碼分析ConcurrentHashMap如何保證線程安全的文章就介紹到這了,更多相關ConcurrentHashMap保證線程安全內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • java求100以內的素數(shù)示例分享

    java求100以內的素數(shù)示例分享

    素數(shù)是指因數(shù)只有1和本身的數(shù)字,這篇文章主要介紹了java求100以內的素數(shù)示例,需要的朋友可以參考下
    2014-03-03
  • 使用Spring MVC攔截器實現(xiàn)日志記錄的方法

    使用Spring MVC攔截器實現(xiàn)日志記錄的方法

    本篇文章主要介紹了使用Spring MVC攔截器實現(xiàn)日志記錄的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • Java設計模式之單例模式簡介

    Java設計模式之單例模式簡介

    這篇文章主要介紹了Java設計模式之單例模式簡介,文中有非常詳細的代碼示例,對正在學習Java的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-04-04
  • Mybatis注解增刪改查的實例代碼

    Mybatis注解增刪改查的實例代碼

    這篇文章主要給大家介紹了關于Mybatis注解增刪改查的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-03-03
  • Java設計模式之抽象工廠模式AbstractFactoryPattern詳解

    Java設計模式之抽象工廠模式AbstractFactoryPattern詳解

    這篇文章主要介紹了Java設計模式之抽象工廠模式AbstractFactoryPattern詳解,抽象工廠模式是一種軟件開發(fā)設計模式,抽象工廠模式提供了一種方式,可以將一組具有同一主題的單獨的工廠封裝起來,需要的朋友可以參考下
    2023-10-10
  • java圖片驗證碼生成教程詳解

    java圖片驗證碼生成教程詳解

    這篇文章主要為大家詳細介紹了java圖片驗證碼生成教程,從簡單到復雜,從本地到前后臺,感興趣的小伙伴們可以參考一下
    2016-07-07
  • Java線程池ThreadPoolExecutor源碼深入分析

    Java線程池ThreadPoolExecutor源碼深入分析

    ThreadPoolExecutor作為java.util.concurrent包對外提供基礎實現(xiàn),以內部線程池的形式對外提供管理任務執(zhí)行,線程調度,線程池管理等等服務
    2022-08-08
  • 淺談使用Maven插件構建Docker鏡像的方法

    淺談使用Maven插件構建Docker鏡像的方法

    本篇文章主要介紹了淺談使用Maven插件構建Docker鏡像的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-12-12
  • Java8的Lambda表達式你真的會嗎

    Java8的Lambda表達式你真的會嗎

    這篇文章主要介紹了Java8的Lambda表達式你真的會嗎,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-01-01
  • netty中pipeline的handler添加刪除分析

    netty中pipeline的handler添加刪除分析

    這篇文章主要為大家介紹了netty中pipeline的handler添加刪除分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04

最新評論