真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

多線程(十五、ConcurrentHashMap原理(2)類和方法分析)

ConcurrentHashMap的構(gòu)造

ConcurrentHashMap,采用了一種“懶加載”的模式,只有到首次插入鍵值對的時候,才會真正的去初始化table數(shù)組。

構(gòu)造方法:

1、空構(gòu)造函數(shù),默認(rèn)桶大小16
多線程(十五、ConcurrentHashMap原理(2)類和方法分析)
2、指定桶初始容量的構(gòu)造器,必須是2次冪值

站在用戶的角度思考問題,與客戶深入溝通,找到桑珠孜網(wǎng)站設(shè)計與桑珠孜網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗,讓設(shè)計與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個性化、用戶體驗好的作品,建站類型包括:成都網(wǎng)站設(shè)計、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣、空間域名、虛擬主機、企業(yè)郵箱。業(yè)務(wù)覆蓋桑珠孜地區(qū)。

/**
 * 指定table初始容量的構(gòu)造器.
 * tableSizeFor會返回大于入?yún)ⅲ╥nitialCapacity + (initialCapacity >>> 1) + 1)的  最小2次冪值
 */
public ConcurrentHashMap(int initialCapacity) {
    if (initialCapacity < 0)
        throw new IllegalArgumentException();

    int cap = ((initialCapacity >= (MAXIMUM_CAPACITY >>> 1)) ? MAXIMUM_CAPACITY :
        tableSizeFor(initialCapacity + (initialCapacity >>> 1) + 1));

    this.sizeCtl = cap;
}

3、根據(jù)已有的Map構(gòu)造
4、指定table初始容量和負(fù)載因子的構(gòu)造器
5、指定table初始容量、負(fù)載因子、并發(fā)級別的構(gòu)造器

常用字段介紹

/**
 * 最大容量.
 */
private static final int MAXIMUM_CAPACITY = 1 << 30;

/**
 * 默認(rèn)初始容量
 */
private static final int DEFAULT_CAPACITY = 16;

/**
 * 負(fù)載因子,為了兼容JDK1.8以前的版本而保留。
 * JDK1.8中的ConcurrentHashMap的負(fù)載因子恒定為0.75
 */
private static final float LOAD_FACTOR = 0.75f;

/**
 * 鏈表轉(zhuǎn)樹的閾值,即鏈接結(jié)點數(shù)大于8時, 鏈表轉(zhuǎn)換為樹.
 */
static final int TREEIFY_THRESHOLD = 8;

/**
 * 樹轉(zhuǎn)鏈表的閾值,即樹結(jié)點樹小于6時,樹轉(zhuǎn)換為鏈表.
 */
static final int UNTREEIFY_THRESHOLD = 6;

/**
 * 在鏈表轉(zhuǎn)變成樹之前,還會有一次判斷:
 * 即只有桶大小數(shù)量大于MIN_TREEIFY_CAPACITY,才會發(fā)生轉(zhuǎn)換。
 * 這是為了避免在Table建立初期,多個鍵值對恰好被放入了同一個鏈表中而導(dǎo)致不必要的轉(zhuǎn)化。
 */
static final int MIN_TREEIFY_CAPACITY = 64;

/**
 * 在樹轉(zhuǎn)變成鏈表之前,還會有一次判斷:
 * 即只有桶的數(shù)量小于MIN_TRANSFER_STRIDE,才會發(fā)生轉(zhuǎn)換.
 */
private static final int MIN_TRANSFER_STRIDE = 16;

/**
 * 用于在擴容時生成唯一的隨機數(shù).
 */
private static int RESIZE_STAMP_BITS = 16;

/**
 * 可同時進行擴容操作的最大線程數(shù).
 */
private static final int MAX_RESIZERS = (1 << (32 - RESIZE_STAMP_BITS)) - 1;

static final int MOVED = -1;                    // 標(biāo)識ForwardingNode結(jié)點
static final int TREEBIN = -2;                 // 標(biāo)識紅黑樹的根結(jié)點
static final int RESERVED = -3;             // 標(biāo)識ReservationNode結(jié)點()
static final int HASH_BITS = 0x7fffffff;    // usable bits of normal node hash

/**
 * CPU核心數(shù),擴容時使用
 */
static final int NCPU = Runtime.getRuntime().availableProcessors();

/**
 * Node數(shù)組,標(biāo)識整個Map,首次插入元素時創(chuàng)建,大小總是2的冪次.
 */
transient volatile Node[] table;

/**
 * 擴容后的新Node數(shù)組,只有在擴容時才非空.
 */
private transient volatile Node[] nextTable;

/**
 * 控制table的初始化和擴容.
 * 0  : 初始默認(rèn)值
 * -1 : 有線程正在進行table的初始化
 * >0 : table初始化時使用的容量,或初始化/擴容完成后的threshold
 * -(1 + nThreads) : 記錄正在執(zhí)行擴容任務(wù)的線程數(shù)
 */
private transient volatile int sizeCtl;

/**
 * 擴容時需要用到的一個下標(biāo)變量.
 */
private transient volatile int transferIndex;

/**
 * 計數(shù)基值,當(dāng)沒有線程競爭時,計數(shù)將加到該變量上。類似于LongAdder的base變量
 */
private transient volatile long baseCount;

/**
 * 計數(shù)數(shù)組,出現(xiàn)并發(fā)沖突時使用。類似于LongAdder的cells數(shù)組
 */
private transient volatile CounterCell[] counterCells;

/**
 * 自旋標(biāo)識位,用于CounterCell[]擴容時使用。類似于LongAdder的cellsBusy變量
 */
private transient volatile int cellsBusy;

put方法

/**
 * 插入鍵值對,均不能為null.
 */
public V put(K key, V value) {
    return putVal(key, value, false);
}
/**
     * 實際的插入操作
     *
     * @param onlyIfAbsent true:僅當(dāng)key不存在時,才插入
     */
    final V putVal(K key, V value, boolean onlyIfAbsent) {
        if (key == null || value == null) throw new NullPointerException();
        int hash = spread(key.hashCode());  // 再次計算hash值

        /**
         * 使用鏈表保存時,binCount記錄table[i]這個桶中所保存的節(jié)點數(shù);
         * 使用紅黑樹保存時,binCount==2,保證put后更改計數(shù)值時能夠進行擴容檢查,同時不觸發(fā)紅黑樹化操作
         */
        int binCount = 0;

        for (Node[] tab = table; ; ) {            // 自旋插入結(jié)點,直到成功
            Node f;
            int n, i, fh;
            if (tab == null || (n = tab.length) == 0)                   // CASE1: 首次初始化table —— 懶加載
                tab = initTable();
            else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {    // CASE2: table[i]對應(yīng)的桶為null
                // 注意下上面table[i]的索引i的計算方式:[ key的hash值 & (table.length-1) ]
                // 這也是table容量必須為2的冪次的原因,讀者可以自己看下當(dāng)table.length為2的冪次時,(table.length-1)的二進制形式的特點 —— 全是1
                // 配合這種索引計算方式可以實現(xiàn)key的均勻分布,減少hash沖突
                if (casTabAt(tab, i, null, new Node(hash, key, value, null))) // 插入一個鏈表結(jié)點
                    break;
            } else if ((fh = f.hash) == MOVED)                          // CASE3: 發(fā)現(xiàn)ForwardingNode結(jié)點,說明此時table正在擴容,則嘗試協(xié)助數(shù)據(jù)遷移
                tab = helpTransfer(tab, f); // 遷移數(shù)據(jù)方法
            else {                                                      // CASE4: 出現(xiàn)hash沖突,也就是table[i]桶中已經(jīng)有了結(jié)點
                V oldVal = null;
                synchronized (f) {              // 鎖住table[i]結(jié)點
                    if (tabAt(tab, i) == f) {   // 再判斷一下table[i]是不是第一個結(jié)點, 防止其它線程的寫修改
                        if (fh >= 0) {          // CASE4.1: table[i]是鏈表結(jié)點
                            binCount = 1;
                            for (Node e = f; ; ++binCount) {
                                K ek;
                                // 找到“相等”的結(jié)點,判斷是否需要更新value值
                                if (e.hash == hash && ((ek = e.key) == key || (ek != null && key.equals(ek)))) {
                                    oldVal = e.val;
                                    if (!onlyIfAbsent)
                                        e.val = value;
                                    break;
                                }
                                Node pred = e;
                                if ((e = e.next) == null) {     // “尾插法”插入新結(jié)點
                                    pred.next = new Node(hash, key,
                                            value, null);
                                    break;
                                }
                            }
                        } else if (f instanceof TreeBin) {  // CASE4.2: table[i]是紅黑樹結(jié)點
                            Node p;
                            binCount = 2;
                            if ((p = ((TreeBin) f).putTreeVal(hash, key, value)) != null) {
                                oldVal = p.val;
                                if (!onlyIfAbsent)
                                    p.val = value;
                            }
                        }
                    }
                }
                if (binCount != 0) {
                    if (binCount >= TREEIFY_THRESHOLD)
                        treeifyBin(tab, i);     // 鏈表 -> 紅黑樹 轉(zhuǎn)換
                    if (oldVal != null)         // 表明本次put操作只是替換了舊值,不用更改計數(shù)值
                        return oldVal;
                    break;
                }
            }
        }
        addCount(1L, binCount);             // 計數(shù)值加1
        return null;
    }  

putVal一共有4種情況

1、首次插入第一個值,初始化table
private final Node[] initTable() {
    Node[] tab;
    int sc;
    while ((tab = table) == null || tab.length == 0) {  //自旋直到初始化成功
        if ((sc = sizeCtl) < 0)         // sizeCtl<0 說明table已經(jīng)正在初始化/擴容
            Thread.yield();
        else if (U.compareAndSwapInt(this, SIZECTL, sc, -1)) {  // 將sizeCtl更新成-1,表示正在初始化中
            try {
                if ((tab = table) == null || tab.length == 0) {
                    int n = (sc > 0) ? sc : DEFAULT_CAPACITY;
                    Node[] nt = (Node[]) new Node[n];
                    table = tab = nt;
                    sc = n - (n >>> 2);     //  0.75n,負(fù)載因子
                }
            } finally {
                sizeCtl = sc;               // 設(shè)置threshold = 0.75 * table.length
            }
            break;
        }
    }
    return tab;
}
2、table[i]對應(yīng)的桶為空,直接占用table[i]
3、ForwardingNode結(jié)點,說明此時table正在擴容,則嘗試協(xié)助進行數(shù)據(jù)遷移
4、table[i]桶中已經(jīng)有了結(jié)點,hash沖突了,有2種情況
4.1 當(dāng)table[i]的結(jié)點類型為Node——鏈表結(jié)點時,就會將新結(jié)點以“尾插法”的形式插入鏈表的尾部。
4.2 當(dāng)table[i]的結(jié)點類型為TreeBin——紅黑樹代理結(jié)點時,就會將新結(jié)點通過紅黑樹的插入方式插入。
/**
 * 嘗試進行 鏈表 -> 紅黑樹 的轉(zhuǎn)換.
 */
private final void treeifyBin(Node[] tab, int index) {
    Node b;
    int n, sc;
    if (tab != null) {

        // CASE 1: table的容量 < MIN_TREEIFY_CAPACITY(64)時,直接進行table擴容,不進行紅黑樹轉(zhuǎn)換
        if ((n = tab.length) < MIN_TREEIFY_CAPACITY)
            tryPresize(n << 1);

            // CASE 2: table的容量 ≥ MIN_TREEIFY_CAPACITY(64)時,進行鏈表 -> 紅黑樹的轉(zhuǎn)換
        else if ((b = tabAt(tab, index)) != null && b.hash >= 0) {
            synchronized (b) {
                if (tabAt(tab, index) == b) {
                    TreeNode hd = null, tl = null;

                    // 遍歷鏈表,建立紅黑樹
                    for (Node e = b; e != null; e = e.next) {
                        TreeNode p = new TreeNode(e.hash, e.key, e.val, null, null);
                        if ((p.prev = tl) == null)
                            hd = p;
                        else
                            tl.next = p;
                        tl = p;
                    }
                    // 以TreeBin類型包裝,并鏈接到table[index]中
                    setTabAt(tab, index, new TreeBin(hd));
                }
            }
        }
    }
}

get方法

/**
 * 根據(jù)key查找對應(yīng)的value值
 *
 * @return 查找不到則返回null
 */
public V get(Object key) {
    Node[] tab;
    Node e, p;
    int n, eh;
    K ek;
    int h = spread(key.hashCode());     // 重新計算key的hash值
    if ((tab = table) != null && (n = tab.length) > 0 &&
            (e = tabAt(tab, (n - 1) & h)) != null) {
        if ((eh = e.hash) == h) {       // CASE1、table[i]就是待查找的項,直接返回
            if ((ek = e.key) == key || (ek != null && key.equals(ek)))
                return e.val;
        } else if (eh < 0)              //CASE2、hash值<0, 說明遇到非鏈表結(jié)點, 調(diào)用對應(yīng)節(jié)點的find方法查找
            return (p = e.find(h, key)) != null ? p.val : null;
        while ((e = e.next) != null) {  //始終可以按照鏈表方式查找
            if (e.hash == h &&
                    ((ek = e.key) == key || (ek != null && key.equals(ek))))
                return e.val;
        }
    }
    return null;
}

對于CASE2,重點看一下TreeBin結(jié)點的查找

1、TreeBin的查找

ConcurrentHashMap采用了一種類似讀寫鎖的方式:當(dāng)線程持有寫鎖(修改紅黑樹)時,如果讀線程需要查找,不會像傳統(tǒng)的讀寫鎖那樣阻塞等待,而是轉(zhuǎn)而以鏈表的形式進行查找(TreeBin本身時Node類型的子類,所有擁有Node的所有字段)
/**
 * 從根結(jié)點開始遍歷查找,找到“相等”的結(jié)點就返回它,沒找到就返回null
 * 當(dāng)存在寫鎖時,以鏈表方式進行查找
 */
final Node find(int h, Object k) {
    if (k != null) {
        for (Node e = first; e != null; ) {
            int s;
            K ek;
            /**
             * 兩種特殊情況下以鏈表的方式進行查找:
             * 1. 有線程正持有寫鎖,這樣做能夠不阻塞讀線程
             * 2. 有線程等待獲取寫鎖,不再繼續(xù)加讀鎖,相當(dāng)于“寫優(yōu)先”模式
             */
            if (((s = lockState) & (WAITER | WRITER)) != 0) {
                if (e.hash == h &&
                    ((ek = e.key) == k || (ek != null && k.equals(ek))))
                    return e;
                e = e.next;     // 鏈表形式
            }

            // 讀線程數(shù)量加1,讀狀態(tài)進行累加
            else if (U.compareAndSwapInt(this, LOCKSTATE, s, s + READER)) {
                TreeNode r, p;
                try {
                    p = ((r = root) == null ? null :
                        r.findTreeNode(h, k, null));
                } finally {
                    Thread w;
                    // 如果當(dāng)前線程是最后一個讀線程,且有寫線程因為讀鎖而阻塞,則喚醒寫線程,嘗試獲取寫鎖
                    if (U.getAndAddInt(this, LOCKSTATE, -READER) == (READER | WAITER) && (w = waiter) != null)
                        LockSupport.unpark(w);
                }
                return p;
            }
        }
    }
    return null;
}

新聞標(biāo)題:多線程(十五、ConcurrentHashMap原理(2)類和方法分析)
標(biāo)題鏈接:http://weahome.cn/article/jsccco.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部