JDK 8 下ConcurrentHashMap 存在的一些问题

-
-
2025-03-01

使用不当造成死循环问题

computeIfAbsent 方法当使用不当时,可能出现死循环

 

最小复现过程

    public static void main(String[]v) throws InterruptedException {
        ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>(16);
        map.computeIfAbsent(
                "AaAa",
                key -> {
                    return map.computeIfAbsent(
                            "BBBB",
                            key2 -> 42);
                }
        );
    } 

 

问题出现的原因

问题在于在进行 computeIfAbsent 的时候,里面还有一个 computeIfAbsent。而这两个 computeIfAbsent 它们的 key 对应的 hashCode 是一样的。

导致后一个computeIfAbsent  因为前一个computeIfAbsent 设置了ReservationNode节点 无法跳出循环

 

源码分析

    public V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction) {
        if (key == null || mappingFunction == null)
            throw new NullPointerException();
        //AaAa时 : 计算hashCode  = 2031775
        //BBBB时: 计算hashCode  = 2031775
        int h = spread(key.hashCode());
        V val = null;
        int binCount = 0;
        // 死循环
        for (ConcurrentHashMap.Node<K,V>[] tab = table;;) {
            ConcurrentHashMap.Node<K,V> f; int n, i, fh;
            //AaAa时 第一次循环的时候 initTable()
            // BBBB 已经不是null 不会进入
            if (tab == null || (n = tab.length) == 0)
                tab = initTable();
            //AaAa时 第二次循环先是计算出数组的下标,并取出该下标的 node。发现这个 node 是空的。于是进入分支判断:
            //BBBB 时 这个node是A放入的ReservationNode节点 所以不会进入
            else if ((f = tabAt(tab, i = (n - 1) & h)) == null) {
                //AaAa时 key 不存在 且 hash 对应的位置还没值  设置ReservationNode节点
                ConcurrentHashMap.Node<K,V> r = new ConcurrentHashMap.ReservationNode<K,V>();
                synchronized (r) {
                    //AaAa时 cas 操作设置值
                    if (casTabAt(tab, i, null, r)) {
                        binCount = 1;
                        ConcurrentHashMap.Node<K,V> node = null;
                        try {
                            //AaAa时 触发BBBB的computeIfAbsent
                            if ((val = mappingFunction.apply(key)) != null)
                                node = new ConcurrentHashMap.Node<K,V>(h, key, val, null);
                        } finally {
                            setTabAt(tab, i, node);
                        }
                    }
                }
                if (binCount != 0)
                    break;
            }
            //MOVED 表示当前的 ConcurrentHashMap 是否是在进行扩容
            // BBBB 时是ReservationNode hash是-3 不会进入
            else if ((fh = f.hash) == MOVED)
                // 扩容
                tab = helpTransfer(tab, f);
            else {
                // BBBB 时 进入这
                boolean added = false;
                // 可以看到,每次的查找都会加锁(synchronized),
                synchronized (f) {
                    //BBBB 时 取出来的对象就是ReservationNode
                    if (tabAt(tab, i) == f) {
                        // BBBB 时 fh就是 当前 node 的 hash 值  -3 所以不会进入
                        if (fh >= 0) {
                            // 在链表中查找 key
                            binCount = 1;
                            for (ConcurrentHashMap.Node<K,V> e = f;; ++binCount) {
                                K ek; V ev;
                                if (e.hash == h &&
                                        ((ek = e.key) == key ||
                                                (ek != null && key.equals(ek)))) {
                                    val = e.val;
                                    break;
                                }
                                ConcurrentHashMap.Node<K,V> pred = e;
                                if ((e = e.next) == null) {
                                    if ((val = mappingFunction.apply(key)) != null) {
                                        added = true;
                                        pred.next = new ConcurrentHashMap.Node<K,V>(h, key, val, null);
                                    }
                                    break;
                                }
                            }
                        }
                        // BBBB 时 不是红黑树存储
                        else if (f instanceof ConcurrentHashMap.TreeBin) {
                            binCount = 2;
                            ConcurrentHashMap.TreeBin<K,V> t = (ConcurrentHashMap.TreeBin<K,V>)f;
                            ConcurrentHashMap.TreeNode<K,V> r, p;
                            if ((r = t.root) != null &&
                                    (p = r.findTreeNode(h, key, null)) != null)
                                val = p.val;
                            else if ((val = mappingFunction.apply(key)) != null) {
                                added = true;
                                t.putTreeVal(h, key, val);
                            }
                        }
                    }
                }
                // BBBB 时 binCount未被修改所以不会进入
                if (binCount != 0) {
                    if (binCount >= TREEIFY_THRESHOLD)
                        treeifyBin(tab, i);
                    if (!added)
                        return val;
                    break;
                }
            }

            // BBBB 无法跳出循环
        }
        if (val != null)
            addCount(1L, binCount);
        return val;
    }

 

JDK后续修复方案

判断完是否是红黑树节点后,再判断一下是否是 ReservationNode 节点,因为这个节点就是个占位节点。如果是,则抛出异常。

 

多线程下性能问题

computeIfAbsent 方法当Key存在时也会加锁

 

JDK后续修复方案

在没有获取锁的情况下,判断第一个节点key和新key是否相等。

规避大部分场景

 

 

参考资料:

JDK-8161372

JDK-8062841

 


目录