当前位置: 首页 > news >正文

深入探讨Java中的LongAdder:使用技巧与避坑指南

文章目录

      • 一、什么是LongAdder?
      • 二、LongAdder的简单使用
        • 示例代码:
      • 三、LongAdder的工作原理
      • 四、LongAdder的常见使用场景
      • 五、使用LongAdder时的注意事项(避坑指南)
        • 1. 不要滥用LongAdder
        • 2. sum()方法与精度问题
        • 3. 避免过度使用reset()
        • 4. 不能用于CAS依赖的场景
      • 六、LongAdder与AtomicLong的区别
      • 七、总结
      • 推荐阅读文章

在高并发编程中,如何高效地计数或累加值是一个常见问题。Java提供了很多工具来应对这些场景,其中 LongAdder 是一个在高并发环境下性能优于 AtomicLong的类。它是如何工作的?我们如何正确使用它,并避免常见的坑?接下来,我们将通过简单易懂的方式,帮助你理解 LongAdder

一、什么是LongAdder?

LongAdder位于java.util.concurrent.atomic包中,是一种用于高效计数的类。它的功能类似于AtomicLong,但设计上更适合在高并发环境下使用。

AtomicLong依赖于底层的**CAS(Compare-And-Swap)**机制,它通过不断重试来保证原子性。然而,在极高并发的场景中,CAS操作可能会频繁失败,导致性能下降。而LongAdder通过将计数分散到多个单独的变量中,并在最后累加,减少了竞争,从而在高并发场景下提升性能。

二、LongAdder的简单使用

LongAdder的使用非常简单,与AtomicLong类似。你可以使用increment()方法进行累加,用sum()来获取总值。

示例代码:
import java.util.concurrent.atomic.LongAdder;public class LongAdderExample {public static void main(String[] args) {// 创建LongAdder实例LongAdder longAdder = new LongAdder();// 执行累加操作longAdder.increment();longAdder.increment();longAdder.add(10); // 增加指定值// 获取当前累加的总值long sum = longAdder.sum();System.out.println("总计数值: " + sum);  // 输出:12// 重置LongAdderlongAdder.reset();System.out.println("重置后总值: " + longAdder.sum());  // 输出:0}
}

三、LongAdder的工作原理

LongAdder 的核心思想是分段累加,即通过将计数分散到多个变量中(称为“槽”或“单元”),每个线程在并发访问时操作不同的单元,减少竞争。最后,当你调用sum()方法时,所有的单元的值会被汇总,得到最终的总值。

这种设计在低竞争时开销较大,因为每次操作需要涉及多个变量,而在高并发时则能大幅减少CAS失败的重试,提高整体性能。

简单理解

  • 在低并发时,AtomicLong表现会更好,因为不需要额外的分段。
  • 在高并发场景下,LongAdder避免了频繁的CAS冲突,能显著提升效率。

四、LongAdder的常见使用场景

LongAdder特别适合用于高并发计数器场景,例如:

  • Web请求统计:记录每秒钟的访问请求数。
  • 日志系统中的日志条目计数:用于记录日志条目在多线程写入的总数。
  • 性能分析工具:高并发系统中某些操作的频次统计。

五、使用LongAdder时的注意事项(避坑指南)

虽然LongAdder的性能在高并发下非常出色,但使用时也有一些注意事项需要小心。

1. 不要滥用LongAdder

LongAdder的优势主要体现在高并发场景下。如果你的应用并发量较低或只是进行简单的累加操作,那么使用AtomicLong更为合适,因为LongAdder在低并发下反而会有更高的开销。

解决方法:评估你的应用场景,如果并发量不高,优先使用AtomicLong,只有在并发量较大时才使用LongAdder

2. sum()方法与精度问题

由于LongAdder的累加操作是分散到多个单元的,sum()方法是对这些单元进行汇总。因此,当你调用sum()时,可能无法得到瞬时的精确值,特别是在多个线程正在同时进行累加操作时。

解决方法:如果你需要在一个时刻获得精确的计数值,LongAdder可能不适合你。对于大多数场景,这种近似值是可以接受的。

3. 避免过度使用reset()

LongAdder提供了reset()方法来重置计数器,但要小心:reset()只是将累加器清零,且不会对每个线程的单元做特殊处理。在高并发的情况下,reset()后的新累加操作可能会受到原先单元状态的影响,导致不一致的行为。

解决方法:尽量避免在并发操作过程中频繁使用reset(),如果必须使用,确保在恰当的时机(如所有操作已完成时)调用。

4. 不能用于CAS依赖的场景

虽然LongAdder在累加操作中表现出色,但它并不支持AtomicLongCAS操作。如果你的应用场景需要进行基于比较的原子性操作(如compareAndSet()),那AtomicLong是你更好的选择。

六、LongAdder与AtomicLong的区别

功能/特性LongAdderAtomicLong
性能高并发场景下性能优于AtomicLong在低并发场景下表现更好
原子性适合累加操作,但不支持compareAndSet()支持compareAndSet()等CAS操作
开销分散到多个单元,低并发时有额外开销简单直接,开销较小
使用场景高并发计数器需要单步原子性操作或低并发计数场景

七、总结

  • LongAdder是什么? 它是AtomicLong的替代品,设计用于高并发环境下的高效计数。
  • 如何使用? 提供了increment()add()sum()等简单的方法,帮助你进行线程安全的累加操作。
  • 避坑指南? 注意避免在低并发下使用LongAdder,小心reset()操作带来的潜在问题,并且LongAdder无法用于需要CAS操作的场景。

通过正确使用LongAdder,你可以在高并发场景下更高效地进行计数操作,但在选择它之前,务必先评估你的需求和场景,确保它是最佳选择。

推荐阅读文章

  • 使用 Spring 框架构建 MVC 应用程序:初学者教程
  • 有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
  • 如何理解应用 Java 多线程与并发编程?
  • Java Spring 中常用的 @PostConstruct 注解使用总结
  • 线程 vs 虚拟线程:深入理解及区别
  • 深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
  • 10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
  • “打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”
  • Java 中消除 If-else 技巧总结
  • 线程池的核心参数配置(仅供参考)
  • 【人工智能】聊聊Transformer,深度学习的一股清流(13)
  • Java 枚举的几个常用技巧,你可以试着用用
  • 如何理解线程安全这个概念?
  • 理解 Java 桥接方法
  • Spring 整合嵌入式 Tomcat 容器
  • Tomcat 如何加载 SpringMVC 组件

相关文章:

深入探讨Java中的LongAdder:使用技巧与避坑指南

文章目录 一、什么是LongAdder?二、LongAdder的简单使用示例代码: 三、LongAdder的工作原理四、LongAdder的常见使用场景五、使用LongAdder时的注意事项(避坑指南)1. 不要滥用LongAdder2. sum()方法与精度问题3. 避免过度使用rese…...

【本科毕业设计】基于单片机的智能家居防火防盗报警系统

基于单片机的智能家居防火防盗报警系统 相关资料链接下载摘要Abstract第1章 绪论1.1课题的背景1.2 研究的目的和意义 第2章 系统总体方案设计2.1 设计要求2.2 方案选择和论证2.2.1 单片机的选择2.2.2 显示方案的选择 第3章 系统硬件设计3.1 整体方案设计3.1.1 系统概述3.1.2 系…...

C语言 动态数据结构的C语言实现单向链表-2

建立一个单向链表 在单向链表中查找节点---查找尾节点 在单向链表中查找节点 --- 查找第 n 个节点 向单向链表中插入一个节点 向单向链表的尾部插入一个节点 向单向链表中某节点后插入一个节点 向单向链表中插入一个节点 删除单向链表中的某一节点 链表 vs 数组 动态数据结构...

Github 2024-10-23C开源项目日报 Top10

根据Github Trendings的统计,今日(2024-10-23统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量C项目10PLpgSQL项目1Redis - 内存数据库和数据结构服务器 创建周期:5411 天开发语言:C协议类型:BSD 3-Clause “New” or “Revised” Licen…...

ubuntu20.04 opencv4.0 /usr/local/lib/libgflags.a(gflags.cc.o): relocation报错解决

在一个只有ubuntu20.04的docker环境中配置opencv4.0.0, 什么库都没有,都要重新安装, 其他的问题在网上都找到了解决方案,唯独这个问题比较棘手: [ 86%] Linking CXX executable …/…/bin/opencv_annotation /usr/bin/ld: /usr/lo…...

android openGL ES详解——混合

一、混合概念 混合是一种常用的技巧,通常可以用来实现半透明。但其实它也是十分灵活的,你可以通过不同的设置得到不同的混合结果,产生一些有趣或者奇怪的图象。混合是什么呢?混合就是把两种颜色混在一起。具体一点,就…...

计网--物理层

目录 物理层的任务 1、常见概念 2、信道极限容量 3、传输介质 (1)导引型传输介质 (2)非导引型传输介质 4、信道复用技术 (1)频分 / 时分 复用 (2)波分复用WDM (…...

算法的学习笔记—数组中的逆序对(牛客JZ51)

😀前言 在算法和数据结构领域,"逆序对"是一个经典问题。它在数组中两个数字之间定义,若前面的数字大于后面的数字,则这两个数字组成一个逆序对。我们要做的就是,给定一个数组,找出数组中所有的逆…...

Golang | Leetcode Golang题解之第498题对角线遍历

题目&#xff1a; 题解&#xff1a; func findDiagonalOrder(mat [][]int) []int {m, n : len(mat), len(mat[0])ans : make([]int, 0, m*n)for i : 0; i < mn-1; i {if i%2 1 {x : max(i-n1, 0)y : min(i, n-1)for x < m && y > 0 {ans append(ans, mat[x…...

什么是全局污染?怎么避免全局污染?

全局污染&#xff08;Global Pollution&#xff09;是指在编程过程中&#xff0c;过度使用全局变量或对象导致命名冲突、代码可维护性下降及潜在错误增加的问题。在 JavaScript 等动态语言中&#xff0c;尤其需要关注全局污染的风险。 全局污染的影响 1. 命名冲突 3. 意外修改…...

C# 串口通信教程

串口通信&#xff08;Serial Communication&#xff09;是一种用于设备之间数据传输的常见方法&#xff0c;通常用于与外部硬件设备&#xff08;如传感器、机器人、微控制器&#xff09;进行通信。在 C# 中&#xff0c;System.IO.Ports 命名空间提供了与串口设备交互的功能&…...

PHP编程基础

PHP&#xff08;Hypertext Preprocessor&#xff0c;超文本预处理器&#xff09;是一种广泛使用的开源服务器端脚本语言&#xff0c;主要用于网页开发&#xff0c;同时也可以进行命令行脚本编写。以下是PHP编程的基础知识&#xff1a; 1. PHP文件结构 PHP文件通常以 .php 为扩…...

TwinCAT3下位机配置EAP通讯传递与接收变量

添加EAP设备 DEVICE中右键选择添加新项&#xff0c;添加EAP&#xff08;EtherCAT Automation Protocal&#xff09;选择Network Variables类型&#xff0c;如下图。 设置网络适配器来激活EAP&#xff0c;在Adapter中选择search&#xff0c;选择网络适配器后确定&#xff0c;…...

近似推断 - 期望最大化(EM)篇

前言 近似推断是统计学和机器学习中一个至关重要的领域&#xff0c;尤其在处理复杂模型和不完全数据时显得尤为重要。期望最大化&#xff08; Expectation Maximization \text{Expectation Maximization} Expectation Maximization&#xff0c;简称 EM \text{EM} EM&#xff0…...

arp欺骗及其实验

ARP欺骗&#xff08;ARP Spoofing&#xff09;是一种网络攻击技术&#xff0c;攻击者通过伪造ARP&#xff08;地址解析协议&#xff09;消息&#xff0c;将其MAC地址与目标IP地址关联&#xff0c;从而实现对网络流量的截获、篡改或重定向。以下是ARP欺骗的详细信息&#xff1a;…...

HDU The Boss on Mars(容斥原理)

题目大意&#xff1a; ACM 有 n 名员工&#xff0c;现在是他们从老板那里拿薪水的时候了。所有员工都从 1 到 n 编号。原因不明&#xff0c;如果员工的工作编号是 k&#xff0c;他今年可以获得 k^4 Mars 美元。所以为 ACM 工作的员工非常富有。 因为员工人数太多&#xff0c;…...

nnUnet 大模型学习笔记(续):训练网络(3d_fullres)以及数据集标签的处理

目录 1. 数据集处理 1.1 实现脚本 1.2 json文件 2. 设置读取路径 2.1 设置路径 2.2 数据集转换 2.3 数据集预处理 2.4 训练&#xff08;3d_fullres) 3. 训练结果展示 关于nnUnet 数据集的处理和环境搭建&#xff0c;参考上文&#xff1a;第四章&#xff1a;nnUnet大模…...

Java中的数据结构与集合源码

目录 一、数据结构 1.1 数据结构概念 1.2 研究对象 1.3 常见存储结构 1.3.1 数组 1.3.2 链表 1.单向链表 2.双向链表 1.3.3 二叉树 1.3.4 栈&#xff08;FILO&#xff0c;先进后出&#xff09; 1.3.5 队列&#xff08;FIFO&#xff0c;先进先出&#xff09; 二、集合…...

Java应用程序的测试覆盖率之设计与实现(三)-- jacoco cli 客户端

一、背景 上文已把覆盖率数据采集好了&#xff0c;并提供远程连接的tcp地址及端口。 jacoco cli文档jacoco cli jar包 jacococli.jar 我下载好了&#xff0c;放在github工程里。 本文主要是介绍如何使用jacoco cli 客户端读取并生成覆盖率报告。 二、使用 1、dump覆盖率统…...

Deepin V23 / 统信UOS 下安装与配置 tftp

几个月前&#xff0c;我将开发系统从 ubuntu 切换到 Deepin&#xff0c;当时写过一篇文章《使用国产操作系统作为开发系统》。几个月下来&#xff0c;没有感觉有什么不适应&#xff0c;Ubuntu 能做的事情&#xff0c;在 Deepin 上都能做。而且有 UOS 应用商店的加持&#xff0c…...

KubeSphere 容器平台高可用:环境搭建与可视化操作指南

Linux_k8s篇 欢迎来到Linux的世界&#xff0c;看笔记好好学多敲多打&#xff0c;每个人都是大神&#xff01; 题目&#xff1a;KubeSphere 容器平台高可用&#xff1a;环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…...

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…...

手游刚开服就被攻击怎么办?如何防御DDoS?

开服初期是手游最脆弱的阶段&#xff0c;极易成为DDoS攻击的目标。一旦遭遇攻击&#xff0c;可能导致服务器瘫痪、玩家流失&#xff0c;甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案&#xff0c;帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...

Ubuntu系统下交叉编译openssl

一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机&#xff1a;Ubuntu 20.04.6 LTSHost&#xff1a;ARM32位交叉编译器&#xff1a;arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

Linux链表操作全解析

Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表&#xff1f;1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...

SciencePlots——绘制论文中的图片

文章目录 安装一、风格二、1 资源 安装 # 安装最新版 pip install githttps://github.com/garrettj403/SciencePlots.git# 安装稳定版 pip install SciencePlots一、风格 简单好用的深度学习论文绘图专用工具包–Science Plot 二、 1 资源 论文绘图神器来了&#xff1a;一行…...

Spring Boot 实现流式响应(兼容 2.7.x)

在实际开发中&#xff0c;我们可能会遇到一些流式数据处理的场景&#xff0c;比如接收来自上游接口的 Server-Sent Events&#xff08;SSE&#xff09; 或 流式 JSON 内容&#xff0c;并将其原样中转给前端页面或客户端。这种情况下&#xff0c;传统的 RestTemplate 缓存机制会…...

el-switch文字内置

el-switch文字内置 效果 vue <div style"color:#ffffff;font-size:14px;float:left;margin-bottom:5px;margin-right:5px;">自动加载</div> <el-switch v-model"value" active-color"#3E99FB" inactive-color"#DCDFE6"…...

Cinnamon修改面板小工具图标

Cinnamon开始菜单-CSDN博客 设置模块都是做好的&#xff0c;比GNOME简单得多&#xff01; 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...