高并发之CAS机制和ABA问题
什么是CAS机制
CAS是英文单词Compare and Swap的缩写,翻译过来就是比较并替换
CAS机制中使用了3个基本操作数:内存地址V,旧的预期值A,要修改的新值B。
看如下几个例子:
package com.example.demo.concurrentDemo; import org.junit.Test; import java.util.concurrent.atomic.AtomicInteger; public class CasTest { private static int count = 0; @Test
public void test1(){
for (int j = 0; j < 2; j++) {
new Thread(() -> {
for (int i = 0; i < 10000; i++) {
count++;
}
}).start();
} try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
} //结果必定 count <= 20000
System.out.println(count);
} @Test
public void test2() {
for (int j = 0; j < 2; j++) {
new Thread(() -> {
for (int i = 0; i < 10000; i++) {
synchronized (this) {
count++;
}
}
}).start();
} try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
//synchronized 类似于悲观锁
//synchronized关键字会让没有得到锁资源的线程进入BLOCKED状态,而后在争夺到锁资源后恢复为RUNNABLE状态
//这个过程中涉及到操作系统用户模式和内核模式的转换,代价比较高
System.out.println(count);
} private static AtomicInteger atoCount = new AtomicInteger(0); @Test
public void test3() {
for (int j = 0; j < 2; j++) {
new Thread(() -> {
for (int i = 0; i < 10000; i++) {
atoCount.incrementAndGet();
}
}).start();
} try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
} //Atomic操作类的底层正是用到了“CAS机制”
System.out.println(atoCount);
} }
CAS 缺点
1) CPU开销过大
在并发量比较高的情况下,如果许多线程反复尝试更新某一个变量,却又一直更新不成功,循环往复,会给CPU带来很到的压力。
这个可以通过看:AtomicInteger.incrementAndGet()源码,可知这是一个无限循环,获取实际值与预期值比较,当相等才会跳出循坏。
2) 不能保证代码块的原子性
CAS机制所保证的知识一个变量的原子性操作,而不能保证整个代码块的原子性。比如需要保证3个变量共同进行原子性的更新,就不得不使用synchronized了。
3) ABA问题
这是CAS机制最大的问题所在。
什么是ABA?先看下面例子:
我们先来看一个多线程的运行场景:
时间点1 :线程1查询值是否为A
时间点2 :线程2查询值是否为A
时间点3 :线程2比较并更新值为B
时间点4 :线程2查询值是否为B
时间点5 :线程2比较并更新值为A
时间点6 :线程1比较并更新值为C
在这个线程执行场景中,2个线程交替执行。线程1在时间点6的时候依然能够正常的进行CAS操作,尽管在时间点2到时间点6期间已经发生一些意想不到的变化, 但是线程1对这些变化却一无所知,因为对线程1来说A的确还在。通常将这类现象称为ABA问题。
ABA发生了,但线程不知道。又或者链表的头在变化了两次后恢复了原值,但是不代表链表就没有变化。
ABA隐患
就像兵法讲的:偷梁换柱、李代桃僵
历史事件:赵氏孤儿
解决ABA问题两种方法:
1、悲观锁思路,加锁;
2、乐观锁思路,通过AtomicStampedReference.class
源码实现,具体看源码:
1. 创建一个Pair类来记录对象引用和时间戳信息,采用int作为时间戳,实际使用的时候时间戳信息要做成自增的,否则时间戳如果重复,还会出现ABA的问题。这个Pair对象是不可变对象,所有的属性都是final的, of方法每次返回一个新的不可变对象。
2. 使用一个volatile类型的引用指向当前的Pair对象,一旦volatile引用发生变化,变化对所有线程可见。
3. set方法时,当要设置的对象和当前Pair对象不一样时,新建一个不可变的Pair对象。
4. compareAndSet方法中,只有期望对象的引用和版本号和目标对象的引用和版本好都一样时,才会新建一个Pair对象,然后用新建的Pair对象和原理的Pair对象做CAS操作。
5. 实际的CAS操作比较的是当前的pair对象和新建的pair对象,pair对象封装了引用和时间戳信息。
Demo:
@Test
public void test4() {
final int timeStamp = atoReferenceCount.getStamp(); new Thread(() -> {
while(true){
if(atoReferenceCount.compareAndSet(atoReferenceCount.getReference(),
atoReferenceCount.getReference()+1, timeStamp, timeStamp + 1)){
System.out.println("11111111");
break;
}
}
},"线程1:").start(); new Thread(() -> {
while(true){
if(atoReferenceCount.compareAndSet(atoReferenceCount.getReference(),
atoReferenceCount.getReference()+1, timeStamp, timeStamp + 1)){
System.out.println("2222222");
break;
}
}
},"线程2:").start(); try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println(atoReferenceCount.getReference());
}
第二个没有执行,因为时间戳不对了。
修改下代码:
@Test
public void test4() {
for (int i = 0; i < 10; i++) {
new Thread(() -> {
boolean f = atoReferenceCount.compareAndSet(atoReferenceCount.getReference(),
atoReferenceCount.getReference() + 1, atoReferenceCount.getStamp(),
atoReferenceCount.getStamp() + 1); System.out.println("线程"+Thread.currentThread()+"result="+f);
}, "线程:"+i).start();
}
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println(atoReferenceCount.getReference());
}
结果:可见线程:0,比较的时候发现时间戳变了,所以没有+1。
demo2:
@Test
public void test5() {
for (int i = 0; i < 4; i++) {
new Thread(() -> {
for (int j = 0; j < 500; j++) {
boolean f = atoReferenceCount.compareAndSet(atoReferenceCount.getReference(),
atoReferenceCount.getReference() + 1, atoReferenceCount.getStamp(),
atoReferenceCount.getStamp() + 1); System.out.println("线程"+Thread.currentThread()+">>j="+j+",result="+f);
}
}, "线程:"+i).start();
}
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println(atoReferenceCount.getReference());
}
有3次比较时间戳发现已经不同
参考:
https://blog.csdn.net/qq_32998153/article/details/79529704
高并发之CAS机制和ABA问题的更多相关文章
- 并发——详细介绍CAS机制
一.前言 今天花了点时间了解了一下JDK1.8中ConcurrentHashMap的实现,发现它实现的主要思想就是依赖于CAS机制.CAS机制是并发中比较重要的一个概念,所以今天这篇博客就来详细介 ...
- 并发之atomicInteger与CAS机制
并发之atomic与CAS自旋锁 通过前几章的讲解我们知道i++这种类似操作是不安全的.针对这种情况,我们可能会想到利用synchronize关键字实现线程同步,保证++操作的原子性,的确这是一种有效 ...
- 深入浅出Java并发包—CAS机制
在JDK1.5之前.Java主要靠synchronized这个关键字保证同步,已解决多线程下的线程不安全问题,但是这会导致锁的发生,会引发一些个性能问题. 锁主要存在一下问题 (1)在多线程竞争下,加 ...
- Java CAS机制详解
CAS目的: 在多线程中为了保持数据的准确性,避免多个线程同时操作某个变量,很多情况下利用关键字synchronized实现同步锁,使用synchronized关键字修可以使操作的线程排队等待运行,可 ...
- CAS机制与自旋锁
CAS(Compare-and-Swap),即比较并替换,java并发包中许多Atomic的类的底层原理都是CAS. 它的功能是判断内存中某个地址的值是否为预期值,如果是就改变成新值,整个过程具有原子 ...
- 什么是CAS机制?(转)
围绕下面四个点展开叙述: 一:什么是CAS机制? 二:Java当中CAS的底层实现 三:CAS的ABA问题和解决方法 四:java8对CAS的优化 一:什么是CAS机制? 我们先看一段代码: 启动两个 ...
- 线程安全之CAS机制详解(分析详细,通俗易懂)
背景介绍:假设现在有一个线程共享的变量c=0,让两个线程分别对c进行c++操作100次,那么我们最后得到的结果是200吗? 1.在线程不安全的方式下:结果可能小于200,比如当前线程A取得c的值为3, ...
- (白话理解)CAS机制
(白话理解)CAS机制 通过一段对话我们来了解cas用意 示例程序:启动两个线程,每个线程中让静态变量count循环累加100次. 最终输出的count结果是什么呢?一定会是200吗? 加了同步锁之后 ...
- 对CAS机制的理解(一)
先看一段代码:启动两个线程,每个线程中让静态变量count循环累加100次. public class CountTest { public static int count = 0; public ...
随机推荐
- 常见ORM框架及JDBC操作工具类
在Java 程序里面去连接数据库,最原始的办法是使用JDBC 的API.我们先来回顾一下使用JDBC 的方式,我们是怎么操作数据库的. // 注册JDBC 驱动 Class.forName(" ...
- webpack4.0.1的坑
webpack4在2月底正式发布后,原来的很多做法不能使用,下面把使用webpack4.0.1过程中,出现的问题,一一记录,也欢迎大家补充,谢谢!团结就是力量,众人拾柴火焰高,加油! 关于webpac ...
- HTML5+Canvas手机拍摄,本地压缩上传图片
最近在折腾移动站的开发,涉及到了一个手机里面上传图片.于是经过N久的折腾,找到一个插件,用法如下: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 ...
- C#中XmlTextWriter读写xml文件详细介绍
XmlTextWriter类允许你将XML写到一个文件中去.这个类包含了很多方法和属性,使用这些属性和方法可以使你更容易地处理XML.为了使用这个类,你必须首先创建一个新的XmlTextWriter对 ...
- Android生命周期例子小解
Activity 从创建到进入运行态所触发的事件 onCreate()-->onStart-->onResume() 从运行态到停止态所触发的事件 onPa ...
- RabbitMQ ——四种ExChange及完整示例
RabbitMQ常用的Exchange Type有fanout.direct.topic.headers这四种,下面分别进行介绍. 这四种类的exchange分别有以下一些属性,分别是: name:名 ...
- python爬虫:抓取下载视频文件,合并ts文件为完整视频
1.获取m3u8文件 2.代码 """@author :Eric-chen@contact :sygcrjgx@163.com@time :2019/6/16 15:32 ...
- mysql的锁
前言 mysql锁的概念参考如下连接: 1.http://blog.csdn.net/u013063153/article/details/53432468 2.http://www.yesky.co ...
- HTML5:Canvas-绘制图形
到本文的最后,你将学会如何绘制矩形,三角形,直线,圆弧和曲线,变得熟悉这些基本的形状.绘制物体到Canvas前,需掌握路径,我们看看到底怎么做. 栅格 在我们开始画图之前,我们需要了解一下画布栅格(c ...
- windows平台搭建Mongo数据库复制集(类似集群)(一)
Replica Sets(复制集)是在mongodDB1.6版本开始新增的功能,它可以实现故障自动切换和自动修复功能成员节点的功能,各个DB之间的数据完全一致,大大降低了单点故障的风险. [] 以上 ...