登录
首页 >  文章 >  java教程

DoubleAdder与LongAccumulator高并发优化技巧

时间:2026-05-06 11:55:06 182浏览 收藏

DoubleAdder与LongAccumulator并非简单的“高性能替代品”,而是通过Striped64分段结构(base + @Contended隔离的Cell数组)将高并发写操作分散到多个物理独立的缓存行,从而从根本上缓解伪共享引发的cache line bouncing;但其威力高度依赖正确使用——必须预热cells避免初始化风暴、仅用纯函数(如a+b或Math.max)作为累加器、理解sum()返回的是弱一致快照而非实时精确值,并警惕浮点精度误差与JVM参数(如-XX:-RestrictContended)对@Contended生效的影响;盲目替换AtomicLong不仅无法解决问题,反而可能因错误的初始化时机、竞争路径退化或强一致性误用而引发性能雪崩或逻辑错误。

怎么利用 DoubleAdder 与 LongAccumulator 在超高并发计数场景下消除缓存行竞争

DoubleAdder 和 LongAccumulator 在超高并发下能显著缓解缓存行竞争,但前提是正确使用——它们本身不自动消除伪共享,而是依赖 @sun.misc.Contended 注解 + 分段 Cell 数组结构来隔离竞争。直接替换 AtomicLongAtomicDouble 并不能“开箱即用”地解决所有问题,关键在初始化时机、线程绑定方式和读取语义。

为什么 DoubleAdder 的 add() 不会引发跨核缓存行失效

根本原因不是它“更快”,而是它把原本集中更新的单个 volatile long 拆成了多个物理隔离的写入点:

  • DoubleAdder 继承自 Striped64,内部有 base 字段 + Cell[] cells 数组
  • 每个 Cell 类都加了 @sun.misc.Contended,JVM 会在该对象前后填充 128 字节(默认),确保其独占一个缓存行(通常 64 字节)
  • 线程通过 ThreadLocalRandom.getProbe() 计算哈希,映射到 cells 中某个槽位,绝大多数更新只发生在自己专属的 Cell.value
  • 不同 CPU 核心写不同 Cell → 不同缓存行 → 避免 false sharing 导致的 cache line bouncing

LongAccumulator 的 accumulatorFunction 如何影响竞争分布

LongAccumulator 的构造函数接受一个 LongBinaryOperator,这个函数不仅决定计算逻辑,还会影响竞争路径:

  • 当调用 accumulate(x) 时,若当前线程映射的 Cell 为空或 CAS 失败,会 fallback 到 casBase:此时执行的是 r = function.applyAsLong(b = base, x),而非简单加法
  • 如果 function 是非幂等或耗时操作(如 (a,b) -> a * b + System.nanoTime()),会导致 casBase 路径变慢,增加 fallback 频率,反而加剧 base 竞争
  • 推荐仅使用纯函数、无副作用、O(1) 时间复杂度的操作,例如 (a,b) -> a + b(等价于 LongAdder)、(a,b) -> Math.max(a,b)(a,b) -> a & b
  • 注意:identity 值参与初始化,但不参与 CAS 比较;若设为非零值(如 100L),sum() 结果恒含该偏移,需业务侧明确知晓

超高并发下必须避开的三个初始化与读取陷阱

即便用了 DoubleAdderLongAccumulator,以下操作仍会瞬间拉垮性能或导致结果错乱:

  • 在多线程环境下首次调用 add()accumulate() 前,未预热 cells 数组:首次竞争触发的 longAccumulate() 内部要获取 cellsBusy 自旋锁 + 初始化数组(长度=2),此时所有线程阻塞等待,形成“初始化风暴”
  • 频繁调用 sum() 并期望强一致性:该方法无锁遍历 cells,返回的是弱一致快照;若在每轮循环中都调用 sum() 做条件判断(如 while (adder.sum() ),会因重复遍历和不可预测的中间态造成逻辑错误或死循环
  • DoubleAdder 用于需要精确浮点累加的场景:它的 sum() 返回 double,但内部 Cell 存储的是 volatile double,不保证 IEEE 754 运算顺序,多次 add(0.1)sum() 可能出现 0.30000000000000004 ——这不是 bug,是设计使然;如需精确小数,应改用 BigDecimal + 锁,或接受误差容忍

真正决定是否消除缓存行竞争的,从来不是类名里有没有 “Adder”,而是线程是否被稳定分散到互不干扰的内存位置。DoubleAdderLongAccumulator 提供了这个能力,但不会替你做哈希均匀性校验、不会阻止你在 sum() 上做同步假设、也不会让 JVM 忽略 -XX:-RestrictContended 的存在——这些都得自己盯住。

以上就是《DoubleAdder与LongAccumulator高并发优化技巧》的详细内容,更多关于的资料请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>