0

0

如何利用Java的DoubleAdder进行浮点数并发累加_性能优化方案

P粉602998670

P粉602998670

发布时间:2026-03-16 12:44:03

|

423人浏览过

|

来源于php中文网

原创

DoubleAdder适合高并发、只写不读的计数场景,如请求量统计、埋点上报;不适用于强一致读取、减法运算或金融计算。

如何利用java的doubleadder进行浮点数并发累加_性能优化方案

DoubleAdder 适合什么场景

它专为高并发、只写不读的计数累加设计,比如统计请求量、埋点上报、指标聚合。不是用来替代 double 计算或做精确金融运算的——它不保证实时一致性,也不支持减法或原子比较。

常见错误现象:DoubleAdder.sum() 在累加过程中反复调用,误以为能拿到“当前精确值”;其实它只是快照,可能漏掉正在写入的分段值。

  • 适用:吞吐优先、允许短暂延迟、写远多于读(如每秒万级 increment)
  • 不适用:需要强一致读取、要频繁做 sum() == x 判断、涉及减法或除法中间态
  • AtomicDouble(非 JDK 原生)比,DoubleAdder 在 contended 场景下性能通常高 3–10 倍

怎么正确初始化和累加

别在循环里反复 new,也别拿它当普通变量传参修改。它内部靠线程本地 cell 分片,初始化开销小,但滥用会破坏分片效果。

使用场景:Web 过滤器统计响应时间总和、Flink/Spark 中的 metrics collector、日志采样计数器。

立即学习Java免费学习笔记(深入)”;

  • 初始化直接 new DoubleAdder(),无需额外配置
  • 累加统一走 add(double x),不要用 increment()(那是整数用的)
  • 避免在 lambda 或 ForkJoinTask 里捕获并复用同一个 DoubleAdder 实例,线程切换可能导致 cell 分配失效
DoubleAdder timerSum = new DoubleAdder();
// 正确:每个线程独立调用
timerSum.add(elapsedMs);
// 错误:试图用 += 操作符(编译不过)
// timerSum += elapsedMs;

sum() 的时机和代价

sum() 是唯一读操作,但它要遍历所有线程本地 cell 并加总,不是 O(1)。高并发下如果频繁调用,反而成为瓶颈,甚至引发 false sharing。

课游记AI
课游记AI

AI原生学习产品

下载

性能影响:在 64 核机器上,100 个活跃线程时,单次 sum() 可能耗时 50–200ns,而 add() 通常在 5–15ns。

  • 只在必要时刻调用,比如定时上报、监控拉取、批次 flush
  • 不要放在 hot path 循环里,尤其不能在每次 HTTP 请求结束时都 sum()
  • 如果需近似值,可考虑缓存 + 定期刷新,比如用 ScheduledExecutorService 每 5 秒更新一次 volatile double cachedSum

和 AtomicDouble、synchronized 对比的坑

很多人想当然认为 “AtomicDouble 更精确,所以更好”,其实错在混淆了语义:AtomicDouble 用 CAS 串行化所有写,争抢激烈时大量失败重试,吞吐暴跌;DoubleAdder 放弃强顺序换高吞吐,是正交设计。

容易踩的坑:

  • DoubleAdder 当作 AtomicDouble 的升级版来用,结果发现 sum() 不稳定,怀疑是 bug
  • 在单线程环境硬套 DoubleAdder,反而比 double 局部变量慢 2–3 倍(无分片收益,纯开销)
  • 没注意精度问题:它底层用 long 存放 double 的 bit 表示,累加极大量后仍存在浮点舍入误差,和普通 double 相加一致,不是额外缺陷

真正关键的是根据压测数据选型:QPS 超 5k 且写占比 >90%,DoubleAdder 几乎总是更优;否则老老实实用 synchronized 块包一个 double 字段更省心。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

108

2025.10.23

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

77

2025.10.23

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

193

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

62

2026.01.05

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

786

2023.08.10

常用的数据库软件
常用的数据库软件

常用的数据库软件有MySQL、Oracle、SQL Server、PostgreSQL、MongoDB、Redis、Cassandra、Hadoop、Spark和Amazon DynamoDB。更多关于数据库软件的内容详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1007

2023.11.02

C++多线程并发控制与线程安全设计实践
C++多线程并发控制与线程安全设计实践

本专题围绕 C++ 在高性能系统开发中的并发控制技术展开,系统讲解多线程编程模型与线程安全设计方法。内容包括互斥锁、读写锁、条件变量、原子操作以及线程池实现机制,同时结合实际案例分析并发竞争、死锁避免与性能优化策略。通过实践讲解,帮助开发者掌握构建稳定高效并发系统的关键技术。

2

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号