突击并发编程JUC系列-JDK1.8 扩展类型 LongAdder
突击并发编程JUC系列演示代码地址:
小伙伴们,大家好,我们又见面了,突击并发编程JUC系列实战JDK1.8 扩展类型马上就要发车了。
JDK 1.8 扩展类型如下
初步了解
前面在讲解AtomicLong
,跟大家提到过longAdder
, AtomicLong
通过CAS
提供了非阻塞的原子性操作,相比使用阻塞算法的同步器来说它的性能已经很好了,但是JDK
开发组并不满足于此。使用AtomicLong
时,在高并发下大量线程会同时去竞争更新同一个原子变量,但是由于同时只有一个线程的CAS
操作会成功,这就造成了大量线程竞争失败后,会通过无限循环不断进行自旋尝试CAS
的操作,而这会白白浪费CPU
资源。
从JDK 8
开始,针对Long
型的原子操作,Java
又提供了LongAdder
、LongAccumulator
;针对Double
类型,Java
提供了DoubleAdder
、DoubleAccumulator
。Striped64
相关的类的继承层次如下所示。
LongAdder
克服了高并发下使用AtomicLong
的缺点。既然AtomicLong
的性能瓶颈是由于过多线程同时去竞争一个变量的更新而产生的,LongAdder
则是把一个变量分解为多个变量,让同样多的线程去竞争多个资源,解决了性能问题。
使用AtomicLong时,是多个线程同时竞争同一个原子变量。图示如下
使用 longAdder 多个线程同时竞争一个原子变量,图示如下
LongAdder
是把一个变量拆成多份,变为多个变量,有点像 ConcurrentHashMap
中 的分段锁**把一个Long
型拆成一个base变量外加多个Cell
,每个Cell
包装了一个Long
型变量。**这样,在同等并发量的情况下,争夺单个变量更新操作的线程量会减少,这变相地减少了争夺共享资源的并发量。
另外,多个线程在争夺同一个Cell
原子变量时如果失败了,它并不是在当前Cell
变量上一直自旋CAS
重试,而是尝试在其他Cell
的变量上进行CAS
尝试,这个改变增加了当前线程重试CAS
成功的可能性。最后,在获取LongAdder
当前值时,是把所有Cell
变量的value
值累加后再加上base
返回的。LongAdder
维护了一个延迟初始化的原子性更新数组(默认情况下Cell
数组是null
)和一个基值变量base
。由于Cells
占用的内存是相对比较大的,所以一开始并不创建它,而是在需要时创建,也就是惰性加载。
案例测试
下面通过 AtomicLong
和 LongAdder
分别对百万雄师求和,为了更好的对分别通过 10 、100 、500个线程并发求和百万雄师数量。
AtomicLong 性能测试
public class AtomicExample10 {
// 并发线程数
public static int requestTotal = 500;
// 求和总数
public static int sumTotal = 1000000;
public static AtomicLong count = new AtomicLong(0);
public static void main(String[] args) throws InterruptedException {
ExecutorService executorService = Executors.newFixedThreadPool(requestTotal);
final CountDownLatch countDownLatch = new CountDownLatch(requestTotal);
long start = System.currentTimeMillis();
for (int i = 0; i < requestTotal; i++) {
executorService.execute(() -> {
add();
countDownLatch.countDown();
});
}
countDownLatch.await();
executorService.shutdown();
System.out.println("count=" + count.get());
System.out.println("耗时:" + (System.currentTimeMillis() - start));
}
private static void add() {
// 针对 sumTotal 求和
for (int j = 0; j < sumTotal; j++) {
count.getAndIncrement();
}
}
}
通过 10 、100 、500 个并发线程测试
// 并发线程数 10
count=10000000
耗时:305
// 并发线程数 100
count=100000000
耗时:2301
// 并发线程数 500
count=500000000
耗时:10865
LongAdder 性能测试
public class AtomicExample11 {
// 请求总数
public static int requestTotal = 100;
public static LongAdder count = new LongAdder();
public static void main(String[] args) throws InterruptedException {
final CountDownLatch countDownLatch = new CountDownLatch(requestTotal);
long start = System.currentTimeMillis();
for (int i = 0; i < requestTotal; i++) {
new Thread(() -> {
try {
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
add();
countDownLatch.countDown();
}).start();
}
countDownLatch.await();
System.out.println("count=" + count);
System.out.println("耗时:" + (System.currentTimeMillis() - start));
}
private static void add() {
count.add(1);
}
}
通过 10 、100 、500 个并发线程测试
// 并发线程数 10
count=10000000
耗时:110
// 并发线程数 100
count=100000000
耗时:375
// 并发线程数 500
count=500000000
耗时:1451
总结
在以上的测试并发数越多 LongAdder
性能越突出,LongAdder
是把一个变量拆成多份,分散到多个变量, 通过内部 cells 数组分担了高并发下多线程同时对一个原子变量进行更新时的竞争量,让多个线程可以同时对 cells 数组里面的元素进行并行的更新操作,其核心实现通过空间来换时间 。
欢迎关注公众号 山间木匠 , 我是小春哥,从事 Java 后端开发,会一点前端、通过持续输出系列技术文章以文会友,如果本文能为您提供帮助,欢迎大家关注、在看、 点赞、分享支持,_我们下期再见!_
- TensorFlow-4: tf.contrib.learn 快速入门
- TensorFlow-5: 用 tf.contrib.learn 来构建输入函数
- 前后端分离之vue2.0+webpack2 实战项目 -- html模板拼接
- 抛弃vue-resource拥抱axios
- TensorFlow-6-TensorBoard 可视化学习
- TensorFlow-7-TensorBoard Embedding可视化
- windows下nginx的安装及使用方法入门
- Scala Turtuial-容器(集合)类型
- Scala Turtuial-基本语法
- C++ Virtual And Pure Virtual Explained
- 搭建分布式Spark计算平台
- 像tomcat容器那样自定义一个 Classloader
- react+redux+webpack教程5
- TensorFlow-9-词的向量表示
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Matlab马尔可夫链蒙特卡罗法(MCMC)估计随机波动率(SV) 模型
- 如何从xml文件创建R语言数据框dataframe
- R语言POT超阈值模型和极值理论EVT分析
- R语言使用灰色关联分析(Grey Relation Analysis,GRA)中国经济社会发展指标
- R语言中的模拟过程和离散化:泊松过程和维纳过程
- R语言Lee-Carter模型对年死亡率建模预测预期寿命
- R语言有极值(EVT)依赖结构的马尔可夫链(MC)对洪水极值分析
- RxSwift 封装 CoreBluetooth(一) 配置
- Golang 操作Excel文件
- 腾讯云TKE-搭建prometheus监控(一)
- Android开发中ProgressDialog简单用法示例
- Android实现拍照及图片裁剪(6.0以上权限处理及7.0以上文件管理)
- Android仿微信调用第三方地图应用导航(高德、百度、腾讯)
- Android数据共享 sharedPreferences 的使用方法
- Android NavigationBar问题处理的方法