高并发之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 ...
随机推荐
- 15.队列Queue的特点以及使用,优先级等
#生产者与消费者模式,模式解释:比如MVC设计模式 ''' 1.队列 (1)特点:先进先出 (2)python2 VS python3 python2:from Queue import queue ...
- 微信小程序(二)--逻辑层与界面层
一.逻辑层与界面层分离 小程序开发框架将我们需要完成的编码,划分成了两种类型的编码:逻辑编码(由JavaScript完成,业务数据供给界面事件处理),界面编码(页面结构WXML,页面样式WXSS,展示 ...
- Java并发编程:Concurrent锁机制解析
Java并发编程:Concurrent锁机制解析 */--> code {color: #FF0000} pre.src {background-color: #002b36; color: # ...
- python------生产者消费者模型 和 管道
一.为什么要使用生产者和消费者? 在线程世界里,生产者就是生产数据的线程,消费者就是消费数据的线程,在多线程开发当中,如果生产者处理速度很快,而消费者处理速度很慢,那么生产者就必须等待消费者处理完,才 ...
- elasticsearch 深入 —— Post Filter后置过滤器
过滤查询以及聚合 A natural extension to aggregation scoping is filtering. Because the aggregation operates i ...
- XILINX FPGA 开发板 XC3S250E 核心板 学习板+12模块
北京太速科技有限公司为广大合作单位特设海外代购业务,主要包括各类板卡.相机.传感器.仪器仪表.专用芯片等.代购业务仅收取基本的手续费. 北京太速科技有限公司在线客服:QQ:448468544 淘宝网站 ...
- vue,一路走来(9)--聊天窗口
闲暇时间,介绍一下我做一个聊天窗口的心得.如图: 首先要考虑的是得判断出是自己的信息还是对方发来的信息,给出如图的布局,切换不同的类. <li class="clearfix" ...
- 没有dockerfile的情况下如何查看docker的镜像信息
前言 参考资料 https://baijiahao.baidu.com/s?id=1564406878758073&wfr=spider&for=pc 很实用的功能哈.. 步骤 1.先 ...
- Kotlin中?和!!的区别
很多同学刚上手使用Kotlin知道它有针对Java NullPointerException的管理,而在Kotlin中?和!!均是和NullPointerException有关系,可他们的区别到底是什 ...
- 【LeetCode】双指针 two_pointers(共47题)
[3]Longest Substring Without Repeating Characters [11]Container With Most Water [15]3Sum (2019年2月26日 ...