余瑜的博客 余瑜的博客
首页
  • 并发
  • 线程池
  • spring
  • maven
  • 其他
  • redis
  • mysql
  • linux
  • zookeeper
  • docker
  • terminal
  • kong插件开发
  • 资料
  • leetCode-简单
  • blog
  • 其他
关于
GitHub (opens new window)
首页
  • 并发
  • 线程池
  • spring
  • maven
  • 其他
  • redis
  • mysql
  • linux
  • zookeeper
  • docker
  • terminal
  • kong插件开发
  • 资料
  • leetCode-简单
  • blog
  • 其他
关于
GitHub (opens new window)
  • 并发

    • JVM
    • Synchronized
    • volatile
    • CAS
      • 简介
      • 锁与共享变量
      • CAS原理
      • ABA问题
        • 问题
        • 解决思路
      • 无锁的效果
    • AQS
    • ReentrantLock
    • Condition介绍及使用
    • Condition源码解析
    • jdk1.7-HashMap
    • jdk1.7-ConcurrentHashMap
  • 线程池

  • spring

  • maven

  • 其他

  • JAVA
  • 并发
余瑜
2019-12-29
目录

CAS

# 简介

CAS 全拼为 Compare And Swap 比较和交换, 也有人理解为 Compare And Set比较和赋值

# 锁与共享变量

  • 加锁是一种悲观的策略,它总是认为每次访问共享资源的时候,总会发生冲突,所以宁愿牺牲性能(时间)来保证数据安全。
  • 无锁是一种乐观的策略,它假设线程访问共享资源不会发生冲突,所以不需要加锁,因此线程将不断执行,不需要停止。一旦碰到冲突,就重试当前操作直到没有冲突为止。
  • 无锁的策略使用一种叫做比较交换的技术(CAS Compare And Swap)来鉴别线程冲突,一旦检测到冲突产生,就重试当前操作直到没有冲突为止。

# CAS原理

执行函数:CAS(V,E,N)

V: 准备要被更新的变量
E: 期望值
N: 表示新值
1
2
3

V是共享变量,我们拿着自己准备的这个E,去跟V去比较,如果E == V ,说明当前没有其它线程在操作,所以,我们把N 这个值 写入对象的 V 变量中。如果 E != V ,说明我们准备的这个E,已经过时了,所以我们要重新准备一个最新的E ,去跟V 比较,比较成功后才能更新 V的值为N。

# ABA问题

# 问题

int a = 0;
Thread t1,t2,t3;
t1: a=1   cas(0,0,1)
t2: a=0   cas(1,1,0)
t3: a=1   cas(0,0,1)
1
2
3
4
5

在上方伪代码的情况下,

  1. 线程t1使用cas将a从0修改为1, 期望值为0(成立)
  2. 线程t2使用cas将a从1修改为0, 期望值为1(成立)
  3. 线程t3使用cas将a从0修改为1, 期望值为0(成立)

可以看到, 在 t3 进行对a进行修改的时候, a的值已经被t1,t2修改过了, 这就是aba问题

# 解决思路

解决办法很简单, 在cas的时候设置进去一个版本号, 用于记录. 每次修改都对版本号进行变更

CAS(V,E, version,N)
int a = 0;
Thread t1,t2,t3;
t1: a=1   cas(0,0,0,1)
t2: a=0   cas(1,1,1,0)
t3: a=1   cas(0,0,2,1)
1
2
3
4
5
6

这样在t3 进行修改的时候发现版本号不对, 修改失败.

# 无锁的效果

  • 如果多个线程同时使用CAS操作一个变量的时候,只有一个线程能够修改成功。其余的线程提供的期望值已经与共享变量的值不一样了,所以均会失败。
  • 由于CAS操作属于乐观派,它总是认为自己能够操作成功,所以操作失败的线程将会再次发起操作,而不是被OS挂起。所以说,即使CAS操作没有使用同步锁,其它线程也能够知道对共享变量的影响。
  • 因为其它线程没有被挂起,并且将会再次发起修改尝试,所以无锁操作即CAS操作天生免疫死锁。
  • 另外一点需要知道的是,CAS是系统原语,CAS操作是一条CPU的原子指令,所以不会有线程安全问题。

转自无锁机制----比较交换CAS (opens new window)

上次更新: 2021/02/20, 19:26:07

← volatile AQS→

Theme by Vdoing | Copyright © 2018-2022 逆光世间 | 备案号: 京ICP备19016086号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式