LongAdder和AtomicLong的对比实验

2024-04-13 20:44

本文主要是介绍LongAdder和AtomicLong的对比实验,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

LongAdder 的核心思想是热点分离,与 ConcurrentHashMap 的设计思想类似:将value值分离成一个数组,当多线程访问时,通过Hash算法将线程映射到数组的一个元素进行操作;而获取最终的value结果时,则将数组的元素求和。最终,通过 LongAdder 将内部操作对象从单个value值“演变”成一系列的数组元素,从而减小了内部竞争的粒度。

package cn.jaa.cas;import cn.jaa.util.Print;
import cn.jaa.util.ThreadUtil;
import org.junit.Test;import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.LongAdder;/*** @Author: Jaa* @Description: LongAdder 和 AtomicLong 的对比测试* @Date 2024/4/13*/
public class LongAdderVSAtomicLongTest {// 每条线程的执行轮数final int TURNS = 1000000000;/*** 使用 AtomicLong 完成 10个线程 每个线程累加1000次*/@Testpublic void testAtomicLong() {// 并发任务数final int TASK_AMOUNT = 10;// 线程池,获取CPU密集型任务线程池ThreadPoolExecutor pool = ThreadUtil.getCpuIntenseTargetThreadPool();// 定义一个原子对象AtomicLong atomicLong = new AtomicLong(0);// 线程同步倒数闩CountDownLatch latch = new CountDownLatch(TASK_AMOUNT);long startTime = System.currentTimeMillis();for (int i = 0; i < TASK_AMOUNT; i++) {// 提交任务pool.submit(() -> {try {for (int j = 0; j < TURNS; j++) {// 执行累加操作atomicLong.incrementAndGet();}} catch (Exception e) {e.printStackTrace();}// 线程执行完毕,倒数闩减一latch.countDown();});}try {// 等待所有线程执行完毕,倒数闩完成所有的倒数操作latch.await();} catch (InterruptedException e) {e.printStackTrace();}float time = (System.currentTimeMillis() - startTime) / 1000F;// 输出统计结果Print.tcfo("运行的时长为:" + time);Print.tcfo("累加结果为:" + atomicLong.get());// [main|LongAdderVSAtomicLongTest.testAtomicLong]:运行的时长为:17.283// [main|LongAdderVSAtomicLongTest.testAtomicLong]:累加结果为:1000000000}/*** 使用 LongAdder 完成 10个线程累加1000万次*/@Testpublic void testLongAdder() {// 并发任务数final int TASK_AMOUNT = 10;// 线程池,获取CPU密集型任务线程池ThreadPoolExecutor pool = ThreadUtil.getCpuIntenseTargetThreadPool();// 定义一个 LongAdder 对象LongAdder longAdder = new LongAdder();// 线程同步倒数闩CountDownLatch latch = new CountDownLatch(TASK_AMOUNT);long startTime = System.currentTimeMillis();for (int i = 0; i < TASK_AMOUNT; i++) {// 提交任务pool.submit(() -> {try {for (int j = 0; j < TURNS; j++) {// 执行累加操作longAdder.add(1);}} catch (Exception e) {e.printStackTrace();}// 线程执行完毕,倒数闩减一latch.countDown();});}// 等待所有线程执行完毕,倒数闩完成所有的倒数操作try {latch.await();} catch (InterruptedException e) {e.printStackTrace();}float time = (System.currentTimeMillis() - startTime) / 1000F;// 输出统计结果Print.tcfo("运行的时长为:" + time);Print.tcfo("累加结果为:" + longAdder.longValue());// [main|LongAdderVSAtomicLongTest.testLongAdder]:运行的时长为:1.191// [main|LongAdderVSAtomicLongTest.testLongAdder]:累加结果为:1000000000}
}

为了进行速度的对比,可以多次运行以上用例多次,每一次运行可以修改 TASK_AMOUNT(次数常量)的值。测试5次,TASK_AMOUNT的值从1000到1?000?000?000,对比出来的速度倍数值如下图(参考,测试时间与计算机配置有关系)

通过对比实验可以看到:当有10个线程总计累加10?000次的时候,AtomicLong的性能更好。随着累加次数的增加,CAS操作的次数急剧增多,AtomicLong的性能急剧下降。从对比实验的结果可以看出,在CAS争用最为激烈的场景下,LongAdder的性能是AtomicLong的8倍。

这篇关于LongAdder和AtomicLong的对比实验的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/901169

相关文章

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

Java实现本地缓存的四种方法实现与对比

《Java实现本地缓存的四种方法实现与对比》本地缓存的优点就是速度非常快,没有网络消耗,本地缓存比如caffine,guavacache这些都是比较常用的,下面我们来看看这四种缓存的具体实现吧... 目录1、HashMap2、Guava Cache3、Caffeine4、Encache本地缓存比如 caff

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

详解MySQL中JSON数据类型用法及与传统JSON字符串对比

《详解MySQL中JSON数据类型用法及与传统JSON字符串对比》MySQL从5.7版本开始引入了JSON数据类型,专门用于存储JSON格式的数据,本文将为大家简单介绍一下MySQL中JSON数据类型... 目录前言基本用法jsON数据类型 vs 传统JSON字符串1. 存储方式2. 查询方式对比3. 索引

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

关于MyISAM和InnoDB对比分析

《关于MyISAM和InnoDB对比分析》:本文主要介绍关于MyISAM和InnoDB对比分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录开篇:从交通规则看存储引擎选择理解存储引擎的基本概念技术原理对比1. 事务支持:ACID的守护者2. 锁机制:并发控制的艺

CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比

《CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比》CSS中的position属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布... css 中的 position 属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布局和层叠关

Linux中的more 和 less区别对比分析

《Linux中的more和less区别对比分析》在Linux/Unix系统中,more和less都是用于分页查看文本文件的命令,但less是more的增强版,功能更强大,:本文主要介绍Linu... 目录1. 基础功能对比2. 常用操作对比less 的操作3. 实际使用示例4. 为什么推荐 less?5.

基于Go语言实现Base62编码的三种方式以及对比分析

《基于Go语言实现Base62编码的三种方式以及对比分析》Base62编码是一种在字符编码中使用62个字符的编码方式,在计算机科学中,,Go语言是一种静态类型、编译型语言,它由Google开发并开源,... 目录一、标准库现状与解决方案1. 标准库对比表2. 解决方案完整实现代码(含边界处理)二、关键实现细