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

news2024/10/23 7:57:15

文章目录

      • 一、什么是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

        // 重置LongAdder
        longAdder.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 组件

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2221434.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++贪心】1536. 排布二进制网格的最少交换次数|1880

本文涉及知识点 C贪心 决策包容性 LeetCode1536. 排布二进制网格的最少交换次数 给你一个 n x n 的二进制网格 grid,每一次操作中,你可以选择网格的 相邻两行 进行交换。 一个符合要求的网格需要满足主对角线以上的格子全部都是 0 。 请你返回使网格满…

QUIC 启动!

掘金地址:https://juejin.cn/post/7428200842229006377 引言 QUIC是什么?明明你每天都在用,明明每天都在timing,难道你不知道吗?啊?不会吧,不会吧。 那就让本文来让你全方位的了解这个协议。 …

word表格跨页后自动生成的顶部横线【去除方法】

Hello World! Its been a long time. 这一年重心放在了科研、做事、追寻新的经历上,事有正事、琐事、幸事、哀事,内心与认知成长了一些,思想成熟了几分,技艺也有若干收获。不管怎样,来打个卡吧,纪念一下&…

爬虫日常实战

爬取美团新闻信息,此处采用两种方法实现: 注意点:因为此处的数据都是动态数据,所以一定要考虑好向下滑动数据包会更新的情况,不然就只能读取当前页即第一页数据,方法一通过更新ajax数据包网址页数&#xf…

【MyBatis】初识MyBatis 构建简单框架

目录 MyBatis前言搭建一个简单的MyBatis创建Maven项目引入必要依赖创建数据表结构创建User实体类创建Mapper接口Mapper层Dao层 创建MyBatis的Mapper映射文件编写测试类传统测试类JUnit测试 MyBatis 介绍:MyBatis是一款半自动的ORM持久层框架,具有较高的…

利用自定义 ref 实现函数防抖

今天来简单介绍一个新的方法,使用自定义 ref 实现函数防抖。 1. 自定义 ref 的来源 自定义 ref 防抖函数来自于前端开发中的两个概念:Vue 的响应式系统 和 数防抖(Debounce)。 1、Vue 响应式系统:Vue 提供了 ref 和…

Python学习的自我理解和想法(20)

#1024程序员节|征文# 学的是b站的课程(千锋教育),跟老师写程序,不是自创的代码! 今天是学Python的第20天,学的内容是面向对象中的私有属性,私有方法,多态,单例计模式。开…

【ubuntu18.04】ubuntu18.04升级cmake-3.29.8及还原系统自带cmake操作说明

参考链接 cmake升级、更新(ubuntu18.04)-CSDN博客 升级cmake操作说明 下载链接 Download CMake 下载版本 下载软件包 cmake-3.30.3-linux-x86_64.tar.gz 拷贝软件包到虚拟机 cp /var/run/vmblock-fuse/blockdir/jrY8KS/cmake-3.29.8-linux-x86_64…

spring源码中的,函数式接口,注解@FunctionalInterface

调用方 /org/springframework/beans/factory/support/AbstractBeanFactory.java:333sharedInstance getSingleton(beanName, () -> {try {return createBean(beanName, mbd, args);}catch (BeansException ex) {// Explicitly remove instance from singleton cache: It mi…

高级的SQL查询技巧有哪些?

成长路上不孤单😊😊😊😊😊😊 【14后😊///C爱好者😊///持续分享所学😊///如有需要欢迎收藏转发///😊】 今日分享关于高级SQL查询技巧方面的相关内容&#xf…

MATLAB人脸考勤系统

MATLAB人脸考勤系统课题介绍 该课题为基于MATLAB平台的人脸识别系统。传统的人脸识别都是直接人头的比对,现实意义不大,没有一定的新意。该课题识别原理为:先采集待识别人员的人脸,进行训练,得到人脸特征值。测试的时…

HomeAssistant自定义组件学习-【一】

#环境准备# 按官方的步骤准备就可以,我是在Windows下使用VS Code开发的,安装了WSL(使用模板创建组件需要在WSL环境下完成) 官方链接:https://developers.home-assistant.io/docs/development_environment 环境准备好…

力扣困难题汇总(14道)

题4(困难): 思路: 找两数组中位数,这个看起来简单,顺手反应就是数第(mn)/2个,这个难在要求时间复杂度为log(mn),所以不能这样搞,我的思路是:每次切割长度为较…

【K8s】Kubernetes 词汇表

微思网络 厦门微思网络 K8S认证工程师(CKA)备考与学习指南https://mp.weixin.qq.com/s/XsEVpU7dKnJDBopynWW3GQ K8S-CKA课程试听:Container 概述 词汇表 此术语表旨在提供 Kubernetes 术语的完整、标准列表。其中包含特定于 Kubernetes 的技术术语以及…

uniapp修改input中placeholder样式

Uniapp官方提供了两种修改的属性方法&#xff0c;但经过测试&#xff0c;只有 placeholder-class 属性能够生效 <input placeholder"请输入手机验证码" placeholder-class"input-placeholder"/><!-- css --> <style lang"scss" s…

redis的zset实现下滑滚动分页查询思路

常规zset查询 我们redis的数据为 我们知道 我们常规查询的话 我们假如 zset 表中 有7个元素&#xff0c;然后我们进行分页查询的话&#xff0c;我们一次查3个元素&#xff0c;然后查出来元素 和元素的分数 我们redis的语法应该这样写 zrevrangebyscore wang 1000 0 withsc…

kotlin实现viewpager

说明:kotlin tablayout viewpager adapter实现滑动界面 效果图 step1: package com.example.flushfragmentdemoimport androidx.appcompat.app.AppCompatActivity import android.os.Bundle import androidx.fragment.app.Fragment import androidx.viewpager2.adapter.…

【uni-app学习-2】

一、跳转 方法&#xff1a;在methods中去定义方法&#xff1a; 上述为直接跳转&#xff0c;但是当你要跳转页面是由多个可切换页面组成比如&#xff1a; 这个页面其实是由两个页面组成&#xff0c;一个主页&#xff0c;一个我的&#xff0c;两个页面 路由配置需要用到toob…

java--多态(详解)

目录 一、概念二、多态实现的条件三、向上转型和向下转型3.1 向上转型3.2 向下转型 四、重写和重载五、理解多态5.1练习&#xff1a;5.2避免在构造方法中调用重写的方法&#xff1a; 欢迎来到权权的博客~欢迎大家对我的博客提出指导这是我的博客主页&#xff1a;点击 一、概念…

EasyExcel自定义下拉注解的三种实现方式

文章目录 一、简介二、关键组件1、ExcelSelected注解2、ExcelDynamicSelect接口&#xff08;仅用于方式二&#xff09;3、ExcelSelectedResolve类4、SelectedSheetWriteHandler类 三、实际应用总结 一、简介 在使用EasyExcel设置下拉数据时&#xff0c;每次都要创建一个SheetWr…