话说CAS
一、前言
cas 一般认为是compare and swap 也可以认为是compare and set
cas涉及三个值
1)P 变量内存地址
2)E 期望值 ,CPU做计算之前拿出来的旧值
3) X 需要设置的新值
原子操作为: 拿出内存地址当前的值A ,比较A == E ? 是 : 设置P内存的值为X 否:结束。。失败
(1) 第一篇 话说synchronized 画过CAS的流程图 咱们再来一张?

(2) CAS面试经常问的一个是ABA 问题 什么是ABA ? 上图

(3) 有人说ABA 不影响啊 我反正期望的值是A 你最后是A就得了呗
这个还要看具体的业务,拿生活中例子来说,银行职员小孙,偷拿了银行100万,
然后去投资赚了20万,最后把100万还回去。 你细品。。 银行能允许吗
(4)ABA 的解决方案 版本 version 怎么解决 ?

二、DEMO
1. CAS 简单使用
假如有一个值 int count ,2个线程 每个线程给count加5000次 1
按道理说 每个人给你5000 你应该有1万块
public class CasTest {
public static int count = 0;
public static void main(String[] args) throws InterruptedException {
new Sub("第一个").start();
new Sub("第二个").start();
TimeUnit.SECONDS.sleep(5);
System.out.println("count="+CasTest.count);
}
}
class Sub extends Thread{
private String name;
public Sub(String name) {
this.name = name;
}
public void run() {
System.out.println(name+"开始+");
for (int i = 0; i < 5000; i++) {
try {
CasTest.count = CasTest.count+1;
} catch (Exception e) {
e.printStackTrace();
}
}
System.out.println(name+"加完了..");
}
}
执行结果:
第一个开始+
第二个开始+
第一个加完了..
第二个加完了..
count=6811
解决方案1 加锁synchronized 或者 lock 都可以
public class CasTest02 {
public static Integer count = 0;
public static void main(String[] args) throws InterruptedException {
new Sub02("第一个").start();
new Sub02("第二个").start();
TimeUnit.SECONDS.sleep(5);
System.out.println("count="+CasTest02.count);
}
}
class Sub02 extends Thread{
private String name;
public Sub02(String name) {
this.name = name;
}
public void run() {
System.out.println(name+"开始+");
for (int i = 0; i < 500; i++) {
try {
// 加锁
synchronized (CasTest02.class) {
CasTest02.count = CasTest02.count+1;
}
} catch (Exception e) {
e.printStackTrace();
}
}
System.out.println(name+"加完了..");
}
}
解决方案2: CAS java自带的原子类 AtomicInteger
读者读到这里可以了解一下LongAdder
public class CasTest03 {
public static AtomicInteger count = new AtomicInteger(0);
public static void main(String[] args) throws InterruptedException {
new Sub03("第一个").start();
new Sub03("第二个").start();
TimeUnit.SECONDS.sleep(5);
System.out.println("count="+CasTest03.count);
}
}
class Sub03 extends Thread{
private String name;
public Sub03(String name) {
this.name = name;
}
public void run() {
System.out.println(name+"开始+");
for (int i = 0; i < 5000; i++) {
// 加锁
CasTest03.count.incrementAndGet();
}
System.out.println(name+"加完了..");
}
}
执行结果:
第一个开始+
第二个开始+
第二个加完了..
第一个加完了..
count=10000
2. ABA 问题
这里简单复现一个ABA问题 可能不是很精确 读者朋友体会意思即可
public class CasTest04 {
public static AtomicInteger count = new AtomicInteger(0);
public static void main(String[] args) throws InterruptedException {
new Thread(()->{
count.compareAndSet(0,1);
count.compareAndSet(1,0);
System.out.println("线程1 把count从0 修改为1 再从1 修改为0 ");
},"线程1").start();
new Thread(()->{
try {
TimeUnit.SECONDS.sleep(1);
// 这里是0 但是已经不是他所希望的那个0 了
count.compareAndSet(0,4);
System.out.println("线程2 把count 从0 修改为4");
} catch (Exception e) {
e.printStackTrace();
}
},"线程2").start();
TimeUnit.SECONDS.sleep(5);
System.out.println("count="+count);
}
}
执行结果:
线程1 把count从0 修改为1 再从1 修改为0
线程2 把count 从0 修改为4
count=4
ABA解决 加版本
public class CasTest05 {
public static AtomicStampedReference<Integer> count = new AtomicStampedReference<>(0,0);
public static void main(String[] args) throws InterruptedException {
new Thread(()->{
try {
// 等1秒 让线程2拿到版本
TimeUnit.SECONDS.sleep(1);
boolean res = count.compareAndSet(
0,
1,
count.getStamp(),
count.getStamp() + 1);
boolean res2 = count.compareAndSet
(1,
0,
count.getStamp(),
count.getStamp() + 1);
System.out.println("线程1 把count从0 修改为1 再从1 修改为0 "
+ ( res2 ? "成功!":"失败!"));
}catch (Exception r){
r.printStackTrace();
}
},"线程1").start();
new Thread(()->{
try {
// 版本
int stamp = count.getStamp();
TimeUnit.SECONDS.sleep(2);
// 这里是0 但是已经不是他所希望的那个0 了 版本已经变了
boolean res = count.compareAndSet(0,4,stamp,stamp+1);
System.out.println("线程2 把count 从0 修改为4"
+ ( res ? " 成功!":" 失败!"));
} catch (Exception e) {
e.printStackTrace();
}
},"线程2").start();
TimeUnit.SECONDS.sleep(5);
System.out.println("count="+count.getReference());
}
}
三、 假装学术讨论
/**
* @author 木子的昼夜
*/
public class CasTest {
public static void main(String[] args) {
// JUC包里的原子类 线程安全的
AtomicInteger ai = new AtomicInteger();
// 加1 并返回
Integer res = ai.incrementAndGet();
System.out.println(res);
// 上边这句话的意思相当于
int i = 0;
i = i+1;
int resi = i;
System.out.println(resi);
}
}
/**
* Atomically increments by one the current value.
* 当前值自动加1
* @return the updated value
* 返回更新了的值
*/
public final int incrementAndGet() {
return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}
// Unsafe.java#getAndAddInt
public final int getAndAddInt(Object obj, long offset, int step) {
int E;
do {
var5 = this.getIntVolatile(obj, offset);
// 这里不一定能一次就成功哦 会执行多次
// 跟一个姑娘表白 一次不成 你就灰溜溜走了 ? 活该单身..
} while(!this.compareAndSwapInt(obj, offset, E, E + step));
return var5;
}
// Unsafe.java#getIntVolatile 这个方法是获 对象obj 内存开始地址 相对偏移位置offset 对应的值
// 说白了 就是获取对象对应字段的值
public native int getIntVolatile(Object obj, long offset);
//
public final native boolean compareAndSwapInt(
Object obj, // 被修改属性的对象
long offset, // 被修改字段相对于当前对象内存首地址偏移量 可以通过他直接去内存拿数据
int E, // 期望值
int X);// 需要设置的新值
// obj设置属性的对象 offset 字段相对于类内存起始位置偏移量 e期望值 x要设置的值
UNSAFE_ENTRY(jboolean, Unsafe_CompareAndSetInt(JNIEnv *env, jobject unsafe, jobject obj, jlong offset, jint e, jint x)) {
// 转换对象格式 jobject- > oop
oop p = JNIHandles::resolve(obj);
if (p == NULL) {
// 获取filed对应的内存地址 对象起始地址+偏移量
volatile jint* addr = (volatile jint*)index_oop_from_field_offset_long(p, offset);
// 把addr内存对应的值 设置为x 前提是内存值要等于e
return RawAccess<>::atomic_cmpxchg(addr, e, x) == e;
} else {
//
assert_field_offset_sane(p, offset);
return HeapAccess<>::atomic_cmpxchg_at(p, (ptrdiff_t)offset, e, x) == e;
}
} UNSAFE_END
// 这里就调用了 atomic_cmpxchg: 系统方法 : 原子比较并交换计数值。
atomic_cmpxchg(void* addr, T compare_value, T new_value) {
if (is_hardwired_primitive<decorators>()) {
const DecoratorSet expanded_decorators = decorators | AS_RAW;
return PreRuntimeDispatch::atomic_cmpxchg<expanded_decorators>
(addr, compare_value, new_value);
} else {
return RuntimeDispatch<decorators, T, BARRIER_ATOMIC_CMPXCHG>::atomic_cmpxchg
(addr, compare_value, new_value);
}
}
再底层就是系统级别的实现了,CPU 实现Atomic ,我也是看文章看得,说是有2中方式
- 使用总线锁 总线就是老大 CPU小c 给总线发一个LOCK信号 总线收到之后 小c就独占共享内存了,其他CPU
就没有使用权限了 ,数据夸缓存行时使用总线锁 这时候不能用缓存锁 - 使用缓存锁 大多数时候 我们只需要保证对某一块内存的操作时原子性即可,缓存锁就是如果内存区域被缓存再处理器的缓存行中,并且操作的时候缓存行被锁定了,那么当处理器计算完回写到内存时,处理器就把缓存行的地址修改了,如果这时有两一个处理器回写数据到缓存行,咦? 失效了。。
下边连接是我准备写文章的目录,如果有读者觉得需要添加,请微信公众直接发消息给我。
https://www.processon.com/view/link/5f57817963768959e2dc7dca
最后附上自己公众号刚开始写 愿一起进步:

注意: 以上文字 仅代表个人观点,仅供参考,如有问题还请指出,立即马上连滚带爬的从被窝里出来改正。
话说CAS的更多相关文章
- Java高性能编程之CAS与ABA及解决方法
Java高性能编程之CAS与ABA及解决方法 前言 如果喜欢暗色调的界面或者想换换界面,可以看看我在个人博客发布的 Java高性能编程之CAS与ABA及解决方法. CAS概念 CAS,全称Compar ...
- 随意看看AtomicInteger类和CAS
最近在读jdk源码,怎么说呢?感觉收获还行,比看框架源码舒服多了,一些以前就感觉很模糊的概念和一些类的用法也清楚了好多,举个很简单的例子,我在读Integer类的时候,发现了原来这个类自带缓存,看看如 ...
- .NET应用和AEAI CAS集成详解
1 概述 数通畅联某综合SOA集成项目的统一身份认证工作,需要第三方系统配合进行单点登录的配置改造,在项目中有需要进行单点登录配置的.NET应用系统,本文专门记录.NET应用和AEAI CAS的集成过 ...
- SuperMap-iServer-单点登录功能验证(CAS)
SuperMap-iServer-单点登录功能验证(CAS) 1.测试目的: 验证SuperMap-iServer使用CAS单点登录的功能是否正常. 2.测试环境: SuperMap-iServer8 ...
- 集成基于CAS协议的单点登陆
相信大家对单点登陆(SSO,Single Sign On)这个名词并不感到陌生吧?简单地说,单点登陆允许多个应用使用同一个登陆服务.一旦一个用户登陆了一个支持单点登陆的应用,那么在进入其它使用同一单点 ...
- 【Java并发编程实战】-----“J.U.C”:CAS操作
CAS,即Compare and Swap,中文翻译为"比较并交换". 对于JUC包中,CAS理论是实现整个java并发包的基石.从整体来看,concurrent包的实现示意图如下 ...
- 多线程同步工具——CAS原子变量
这是我参考的一篇文章<基于CAS的乐观锁实现>,讲述的是一种需要CPU支持的执行技术CAS(Compare and Swap). 首先理解什么是原子性操作,意思是不能再拆分的操作,例如改写 ...
- CAS FOR WINDOW ACTIVE DIRECTORY SSO单点登录
一.CAS是什么? CAS(Central Authentication Service)是 Yale 大学发起的一个企业级的.开源的项目,旨在为 Web 应用系统提供一种可靠的单点登录解决方法(支持 ...
- CAS Client集群环境的Session问题及解决方案
[原创申明:文章为原创,欢迎非盈利性转载,但转载必须注明来源] 之前写过一篇文章,介绍单点登录的基本原理.这篇文章重点介绍开源单点登录系统CAS的登录和注销的实现方法.并结合实际工作中碰到的问题,探讨 ...
随机推荐
- 如何使用Gephi工具进行可视化复杂网络图
在Gephi安装官网中也介绍了一些如何使用该工具的方法,我将根据自己的数据和可视化的图片进行介绍 第一步:整理数据格式,我的数据是.csv格式的(.xlsx,.xls等等) 数据第一行第一列必须是相同 ...
- [Golang]-5 协程、通道及其缓冲、同步、方向和选择器
目录 协程 通道 通道缓冲 通道同步 通道方向 通道选择器 协程 Go 协程 在执行上来说是轻量级的线程. 代码演示 import ( "fmt" "time" ...
- C# 数据类型(2)
String char的集合 string name = "John Doe"; 双引号,char是单引号string是不可变的,一旦初始化后就不能变了,每次对已存在的string ...
- STM32 单片机的USART的奇偶校验 误区(坑)
当STM32的串口配置成带有奇偶校验位的情况下,需要软件校验是否发生奇偶校验错误,硬件只是置起奇偶校验错误标志位,并将错误的数据放到DR寄存器中,同时置起RXEN标志位,如果使能中断还是会正常进入中断 ...
- 蓝湖 UI 设计稿上如何生成渐变色和复制渐变色
蓝湖 UI 设计稿上如何生成渐变色和复制渐变色 Sketch 生成渐变色 不要上传图片,切图 如果是切图,切图模式下就不会生成 css 代码了 复制渐变色 OK .button { width: 28 ...
- popstate 事件 & history API
popstate 事件 & history API URL change 当用户浏览会话历史记录时,活动历史记录条目发生更改时,将触发 Window 界面的 popstate 事件. 它将当前 ...
- YouTube 视频下载工具
YouTube 视频下载工具 我不生产视频,只是优秀视频的搬运工! YouTube-dl https://github.com/search?q=youtube-dl https://github.c ...
- 微信小程序 HTTP API
微信小程序 HTTP API promise API https://www.npmtrends.com/node-fetch-vs-got-vs-axios-vs-superagent node-f ...
- vue & components & props & methods & callback
vue & components & props & methods & callback demo solution 1 & props & data ...
- how to close macos eject icon from menu bar
how to close macOS eject icon from the menu bar close eject https://apple.stackexchange.com/question ...