JUC包-原子类(AtomicInteger为例)
JUC包-原子类
为什么需要JUC包中的原子类
首先,一个简单的i++可以分为三步:
读取i的值
计算i+1
将计算出i+1赋给i
这就无法保证i++的原子性,即在i++过程中,可能会出现其他线程也读取了i的
值,但读取到的不是更改过后的i的值。
原子类原理(AtomicInteger为例)
原子类的原子性是通过volatile + CAS实现原子操作的。
volatile

AtomicInteger类中的value是有volatile关键字修饰的,这就保证了value的内存可见性,这为后续的CAS实现提供了基础。
CAS
通过查看源码可以发现,AtomicInteger类的值更新操作都是通过调用
getAndAddInt(Object var1, long var2, int var4)方法实现
/**
* Atomically adds the given value to the current value.
*
* @param delta the value to add
* @return the previous value
*/
public final int getAndAdd(int delta) {
//返回的是修改前的值,类似于i++
return unsafe.getAndAddInt(this, valueOffset, delta);
}
/**
* Atomically adds the given value to the current value.
*
* @param delta the value to add
* @return the updated value
*/
public final int addAndGet(int delta) {
//返回的是更新后的值,类似于++i
return unsafe.getAndAddInt(this, valueOffset, delta) + delta;
}
当我们查看getAndAddInt方法的具体实现,可以发现在整个方法中存在一个循
环,这就是我们说的自旋锁,顾名思义,while语句里面的条件一直为true,这个
循环就会一直执行下去。
public final int getAndAddInt(Object var1, long var2, int var4) {
int var5;
do {
var5 = this.getIntVolatile(var1, var2);
} while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
return var5;
}
下面我们来分析getAndAddInt方法中的各个参数的具体含义:
Object var1:this,表示当前对象
long var2:valueOffset,表示当前对象的内存偏移量
int var4:delat,需要加上的数值
所以整个方法的运行流程可以归纳为:
读取传入对象this在主存中偏移量为offset位置的值赋值给var5
将var5的值与当前线程对象内存中偏移量为offset位置的值进行比较(compare)
如果相等,将var5+var4的值更新到对象内存中偏移量为offset位置(swap);如果不
相等,就进入while循环自旋。
CAS的缺点
循环时间长,开销大
在并发量比较高的情况下,如果许多线程反复尝试更新某一个变量,却
又一直更新不成功,循环往复,会给CPU带来很大的压力
只能保证一个共享变量的原子性操作
CAS机制所保证的只是一个变量的原子性操作,而不能保证整个代码块
的原子性。比如需要保证3个变量共同进行原子性的更新,就不得不使用
Synchronized了
ABA问题
见下文
ABA问题
什么是ABA问题
简单来说就是CAS过程只在乎当前值与期望值是否相等,只在乎最终结果,不考虑中
间变化,具体可以看下面一个简单的例子。
public class Test {
static AtomicInteger atomicInteger = new AtomicInteger(0);
public static void main(String[] args) {
atomicInteger.compareAndSet(0,1);
System.out.println("线程A第一次修改:0->" + atomicInteger.get());
new Thread(() -> {
atomicInteger.compareAndSet(1,0);
System.out.println("线程A第二次修改:1->" + atomicInteger.get());
}, "testA").start();
new Thread(() -> {
try {
//确保A线程修改完毕
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
atomicInteger.compareAndSet(0,2);
System.out.println("线程B第一次修改:0->" + atomicInteger.get());
}, "testB").start();
}
}
程序运行后输出的结果,由此可见AtomicInteger的CAS中间步骤有变化,但是没有被感知到。

ABA问题的解决办法
一个简单的想法是,在数据上加上时间戳(版本号),使得线程每次对变量进行修改时,不仅要对比值,还要
对比时间戳(版本号),每次修改操作都会导致时间戳(版本号)改变为新的
值;
我们通过AtomicStampedReference类引入版本号,如下图所示
public class Test {
//初始化数值为0,版本号为1
static AtomicStampedReference<Integer> atomicStampedReference = new AtomicStampedReference<>(0, 1);
public static void main(String[] args) {
new Thread(() -> {
/* compareAndSet四个参数分别为
* 期望值/新的值/期望版本号/新的版本号
*/
atomicStampedReference.compareAndSet(0, 1, 1, 2);
System.out.println("数值第一次修改为" + atomicStampedReference.getReference() +
" 版本号第一次修改为" + atomicStampedReference.getStamp());
atomicStampedReference.compareAndSet(1, 0, 2, 3);
System.out.println("数值第二次修改为" + atomicStampedReference.getReference() +
" 版本号第二次修改为" + atomicStampedReference.getStamp());
}, "testA").start();
new Thread(() -> {
try {
//确保A线程修改完毕
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
atomicStampedReference.compareAndSet(0, 1, 1, 2);
System.out.println("数值第三次修改为" + atomicStampedReference.getReference() +
" 版本号第三次修改为" + atomicStampedReference.getStamp());
}, "testB").start();
}
}
上述代码的程序运行结果如下图所示,可以看到当第三次修改的时候,虽然期望值0匹配,但是期望版本号不匹配,导致第三次修改无效。

JUC包-原子类(AtomicInteger为例)的更多相关文章
- juc原子类之二:基本类型原子类AtomicInteger(AtomicLong、AtomicBoolean)
一.AtomicInteger简介 AtomicInteger, AtomicLong和AtomicBoolean这3个基本类型的原子类的原理和用法相似.以AtomicInteger对基本类型的原子类 ...
- JUC之原子类
在分析原子类之前,先来了解CAS操作 CAS CAS,compare and swap的缩写,中文翻译成比较并交换. CAS 操作包含三个操作数 —— 内存位置(V).预期原值(A)和新值(B).如果 ...
- 对Java原子类AtomicInteger实现原理的一点总结
java原子类不多,包路径位于:java.util.concurrent.atomic,大致有如下的类: java.util.concurrent.atomic.AtomicBoolean java. ...
- Java原子类AtomicInteger实现原理的一点总结
java原子类不多,包路径位于:java.util.concurrent.atomic,大致有如下的类: java.util.concurrent.atomic.AtomicBoolean java. ...
- java的原子类 AtomicInteger 实现原理是什么?
采用硬件提供原子操作指令实现的,即CAS.每次调用都会先判断预期的值是否符合,才进行写操作,保证数据安全. CAS机制 CAS是英文单词Compare And Swap的缩写,翻译过来就是比较并替换. ...
- CAS你知道吗?原子类AtomicInteger的ABA问题谈谈?
(1)CAS是什么? 比较并交换 举例1, CAS产生场景代码? import java.util.concurrent.atomic.AtomicInteger; public class CA ...
- 浅析CAS与AtomicInteger原子类
一:CAS简介 CAS:Compare And Swap(字面意思是比较与交换),JUC包中大量使用到了CAS,比如我们的atomic包下的原子类就是基于CAS来实现.区别于悲观锁synchroniz ...
- 从原子类和Unsafe来理解Java内存模型,AtomicInteger的incrementAndGet方法源码介绍,valueOffset偏移量的理解
众所周知,i++分为三步: 1. 读取i的值 2. 计算i+1 3. 将计算出i+1赋给i 可以使用锁来保持操作的原子性和变量可见性,用volatile保持值的可见性和操作顺序性: 从一个小例子引发的 ...
- JUC源码学习笔记4——原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法
JUC源码学习笔记4--原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法 volatile的原理和内存屏障参考<Java并发编程的艺术> 原子类源码基于JDK8 ...
随机推荐
- PyQt(Python+Qt)学习随笔: QDoubleSpinBox浮点数字设定部件简介
专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 老猿学5G博文目录 在<PyQt(Python+Qt)学习随笔: ...
- PyQt(Python+Qt)学习随笔:使用QFontDialog.getFont交互设置应用或部件的字体
专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 PyQt中的部件只要是QWidget的派生类都可以在Designer或 ...
- 第8.30节 重写Python __setattr__方法实现属性修改捕获
一. 引言 在<第8.26节 重写Python类中的__getattribute__方法实现实例属性访问捕获>章节介绍了__getattribute__方法,可以通过重写该方法,截获所有通 ...
- 第11.13节 Python正则表达式的转义符”\”功能介绍
为了支持特殊元字符在特定场景下能表示自身而不会被当成元字符进行匹配出来,可以通过字符集或转义符表示方法来表示,字符集表示方法前面在<第11.4节 Python正则表达式搜索字符集匹配功能及元字符 ...
- PyQt(Python+Qt)学习随笔:Designer中PushButton按钮flat属性
flat属性用于确认按钮边框是否凸起,如果为False则凸起,如果为True则边框与背景是平坦的. 默认值为False,如果设置为True,则除非按下按钮,否则大多数样式都不会绘制按钮背景.通过使用s ...
- PHP代码审计分段讲解(2)
03 多重加密 源代码为: <?php include 'common.php'; $requset = array_merge($_GET, $_POST, $_SESSION, $_COOK ...
- 用Ubuntu和树莓派系统部署kubernetes集群后的一些心得
方案 环境 操作系统:Ubuntu 16.04 & Raspbian GNU/Linux 9(Stretch Desktop) kubernetes :1.15.3 flannel:0.11. ...
- SseEmitter推送
后端代码SseController.java package com.theorydance.mywebsocket.server; import java.util.HashMap; import ...
- Python编码相关
1.#coding=utf-8的作用 作用是这个文件代码的编码格式,如果没有声明代码中不能出现中文字符,包括注释中也不能出现.否则会报错SyntaxError: Non-ASCII character ...
- Python轻松入门到项目实战-实用教程
本课程完全基于Python3讲解,针对广大的Python爱好者与同学录制.通过本课程的学习,可以让同学们在学习Python的过程中少走弯路.整个课程以实例教学为核心,通过对大量丰富的经典实例的讲解.让 ...