CAS(全称为CompareAndSwap,也有说是CompareAndSet,都差不多)是一条CPU并发原语,它的功能是判断内存某个位置的值是否为预期值,如果是则更改为新的值,判断预期值和更改新值的整个过程是原子的。在JAVA中,CAS的实现全部在sun.misc.Unsafe类中的各个方法,调用UnSafe类中的CAS方法,JVM会帮我们实现出CAS汇编指令,这是一种完全依赖于硬件的功能。

在传统方式中实现并发的手段是加锁,JAVA中的锁有synchronized和Lock(jdk1.5才有)。Lock是基于AQS和CAS实现的,这里先跳过。对于synchronized锁,JVM在执行它的时候会依赖操作系统的临界区机制。这样的话,每次执行到synchronized锁,都会经历用户态和内核态之间的切换。这个过程的消耗是很大的。而且,大多数时候synchronized锁住的操作是很细粒度的。为了细粒度的操作去经历用户态和内核态之间的切换是低效的做法。

其实最常见的就是我们需要并发修改某个变量值,举个常见的例子,窗口售票,不加锁的代码如下所示:

public class Test {

    public static void main(String[] args) {
Stock stock = new Stock(10);
for (int i = 0; i < 5; i++) {
new Thread(stock, "窗口" + (++i)).start();
}
} private static class Stock implements Runnable {
private volatile int count; public Stock(int count) {
this.count = count;
} @Override
public void run() {
for (;;) {
count--;
if (count < 0) {
return;
}
System.out.println(Thread.currentThread().getName() + "售出一张票,剩余票数:" + count);
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}

得到的结果:
(票数和线程太少,需要多跑几遍,多了几乎一遍就可以看出问题)

得到的结果很明显,一共卖出了12张票,其中红框中出现的数字就可以说明问题。出现问题的原因很简单,因为--count这个算式表达式并不是原子的。在一个线程对count进行计算赋值后,但还没有将新值推送到内存中时,另一个线程获取的count值还是原来的值,当这个线程拿着这个值去进行计算,就会出现上面的问题。(这个涉及到Java的内存模型JMM,有兴趣的可以自行了解)
在JDK1.5之前,我们想要解决这个问题,就只能使用synchronized进行加锁,如下:

public void run() {
for (;;) {
synchronized (this) {
count--;
}
if (count < 0) {
return;
}
System.out.println(Thread.currentThread().getName() + "售出一张票,剩余票数:" + count);
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

但是如果仅仅是类似于--count这种并发计数功能,需要进行同步的操作粒度很细时,使用synchronized就大材小用了,不高效(即便现在synchronized经过很多优化,不再想最初那样耗资源,但是它毕竟是个锁,而且多个线程进行竞争的时候还是会变成重量级锁),而使用CAS来实现就会更加的轻量级,性能更好。先上代码再说

public class Test {

    private static Unsafe unsafe;

    public static void main(String[] args) throws Exception {
Field theUnsafeField = Unsafe.class.getDeclaredField("theUnsafe");
theUnsafeField.setAccessible(true);
unsafe = (Unsafe) theUnsafeField.get(null);
Stock stock = new Stock(10);
for (int i = 0; i < 5; i++) {
new Thread(stock, "窗口" + (++i)).start();
}
} private static class Stock implements Runnable {
private volatile int count;
private static long countOffset;
public Stock(int count) {
this.count = count;
try {
countOffset = unsafe.objectFieldOffset(this.getClass().getDeclaredField("count"));
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
} @Override
public void run() {
for (;;) {
int x = this.count;
int y = x - 1;
if (!unsafe.compareAndSwapInt(this, countOffset, x, y)) {
continue;
}
if (y < 0) {
return;
}
System.out.println(Thread.currentThread().getName() + "售出一张票,剩余票数:" + y);
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}

像上面代码一样,当我们使用CAS进行操作时,出现的效果和使用synchronized一样的。代码中使用了JAVA提供的sun.misc.Unsafe进行CAS操作,而且在代码总我使用反射进行获取Unsafe实例,之所以这样做,是因为JDK不想让我们开发者去直接使用Unsafe这个类,而且使用起来比较繁琐,他们给我们提供了一些封装好的类来供我们开发者使用,比如常用的java.util.concurrent.atomic.AtomicInteger、java.util.concurrent.atomic.AtomicBoolean、java.util.concurrent.atomic.AtomicIntegerArray。这些类中都有相同的特点,就是使用sun.misc.Unsafe进行CAS操作,内部进行了一些类似上面代码的封装,我们就以AtomicInteger进行代码演示。

public class Test {

    public static void main(String[] args) throws Exception {
Stock stock = new Stock(10);
for (int i = 0; i < 5; i++) {
new Thread(stock, "窗口" + (++i)).start();
}
} private static class Stock implements Runnable {
private volatile AtomicInteger count;
public Stock(int count) {
this.count = new AtomicInteger(count);
} @Override
public void run() {
for (;;) {
int x = count.get();
int y = x - 1;
if (!count.compareAndSet(x, y)) {
continue;
}
if (y < 0) {
return;
}
System.out.println(Thread.currentThread().getName() + "售出一张票,剩余票数:" + y);
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}

就像上面的代码一样,我们不必进行反射获取Unsafe,然后在获取字段在class中的偏移量这些繁琐的操作了,下面我们就去看看AtomicInteger的源码

  

看上面AtomicInteger的源码我们可以很明显的看的出来,在AtomicInteger进行类加载的时候,会通过sun.misc.Unsafe获取value这个变量在类文件中的偏移量,进行保存,跟我们直接使用Unsafe的操作是一样的。我们找到刚才使用的compareAndSet(x, y)方法的源码,可以看到底层就是使用unsafe实例进行CAS操作。

AtomicInteger还有一些别的方法,比如getAndIncrement、getAndDecrement、getAndAdd、incrementAndGet、decrementAndGet等等,底层实际上还是使用的unsafe实例进行CAS操作,有兴趣的同学可以自己翻下源码看看,这里就不多说了。

总结:CAS的出现就是为了解决一些简单的并发操作,将比较、赋值作为一个原子操作记性处理,实现无锁化处理,节省资源开销。

JUC之CAS的更多相关文章

  1. 一段JAVA代码了解多线程,JUC、CAS原子性操作。

    @Test public void testPaceController_multiThread() throws InterruptedException { final PaceControlle ...

  2. JUC

    1.Java JUC简介 在Java5.0提供了java.util.concurrent(简称JUC)包,在此包中增加了在并发编程中很常用的实用工具类,用于定义类似于线程的自定义子系统,包括线程池.异 ...

  3. juc-2.1-模拟CAS算法

    package com.wf.zhang.juc; /* * 模拟 CAS 算法 */ public class TestCompareAndSwap { public static void mai ...

  4. volatile关键字与内存可见性&原子变量与CAS算法

    1 .volatile 关键字:当多个线程进行操作共享数据时, 可以保证内存中的数据可见 2 .原子变量:jdk1.5后java.util.concurrent.atomic 包下提供常用的原子变量 ...

  5. Spring Cloud--尚硅谷2020最新版

    Spring Cloud 初识Spring Cloud与微服务 在传统的软件架构中,我们通常采用的是单体应用来构建一个系统,一个单体应用糅合了各种业务模块.起初在业务规模不是很大的情况下,对于单体应用 ...

  6. java多线程系列 JUC原子类 CAS及原子类

    根据数据类型,可以将JUC包中的原子操作类可以分为4类. 1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean ;2. 数组类型: AtomicInteg ...

  7. 【Java_多线程并发编程】JUC原子类——原子类中的volatile变量和CAS函数

    JUC中的原子类是依靠volatile变量和Unsafe类中的CAS函数实现的. 1. volatile变量的特性 内存可见性(当一个线程修改volatile变量的值后,另一个线程就可以实时看到此变量 ...

  8. java并发编程(十三)----(JUC原子类)引用类型介绍(CAS和ABA的介绍)

    这一节我们将探讨引用类型原子类:AtomicReference, AtomicStampedRerence, AtomicMarkableReference.AtomicReference的使用非常简 ...

  9. JUC原子操作类与乐观锁CAS

    JUC原子操作类与乐观锁CAS ​ 硬件中存在并发操作的原语,从而在硬件层面提升效率.在intel的CPU中,使用cmpxchg指令.在Java发展初期,java语言是不能够利用硬件提供的这些便利来提 ...

随机推荐

  1. 数据结构和算法(Golang实现)(8.2)基础知识-分治法和递归

    分治法和递归 在计算机科学中,分治法是一种很重要的算法. 字面上的解释是分而治之,就是把一个复杂的问题分成两个或更多的相同或相似的子问题. 直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合 ...

  2. 自动补全、回滚!介绍一款可视化 sql 诊断利器

    Yearning简介 ================= Yearning MYSQL 是一个SQL语句审核平台.提供查询审计,SQL审核等多种功能,支持Mysql,可以在一定程度上解决运维与开发之间 ...

  3. 用threejs 实现3D物体在浏览器展示

    用threejs 实现3D物体在浏览器展示,通过鼠标平移,缩放,键盘箭头按钮左右移动等功能展示. <!DOCTYPE html> <html> <head> < ...

  4. DeepinV20系统文件管理器右键发送至为知笔记

    1. 创作背景 昨天在深度系统上做了一个打开文件管理器选择文件右键发送文本至博客园的插件. 这个插件对于我自己来说是及其方便的东西,平时的学习积累,工作经验或者生活感悟,随手记下之后,就能够轻松发送出 ...

  5. PHP 将字符串转换为字符集格式UTF8/GB2312/GBK 函数iconv()

     iconv()介绍 iconv函数可以将一种已知的字符集文件转换成另一种已知的字符集文件 iconv('要转化的格式',‘转化后的格式’,‘转化的数据’); 但是转化是经常出错,一般需要在转成的编码 ...

  6. Java 中正则表达式使用

    正则表达式基本用法: 测试代码: @Test public void test01() { String str = "adsfd##4324"; // 创建正则表达式对象 Pat ...

  7. 高性能的JavaScript,这是一个高级程序员必备的技能

    不知道大家有没有看过高性能JavaScript,这个书是一本好书,推荐有JavaScript的基础的同学可以看一看这本书. 下面是我根据这本书整理出来的知识: 1.将经常使用的对象成员.数组项.和域外 ...

  8. 1、flink介绍,反压原理

    一.flink介绍  Apache Flink是一个分布式大数据处理引擎,可对有界数据流和无界数据流进行有状态计算. 可部署在各种集群环境,对各种大小的数据规模进行快速计算. 1.1.有界数据流和无界 ...

  9. [机器学习实战-Logistic回归]使用Logistic回归预测各种实例

    目录 本实验代码已经传到gitee上,请点击查收! 一.实验目的 二.实验内容与设计思想 实验内容 设计思想 三.实验使用环境 四.实验步骤和调试过程 4.1 基于Logistic回归和Sigmoid ...

  10. MySql -- 数据结构

    现在的数据表不单单只是存储数据,还有的是设计功能和快速处理数据的结构功能: 首先,我们在设计数据库的时候,我们要先分清楚,那些是要单纯的存储数据的(固定),然后再设计出来数据的表(流动) 你懂我意思吧 ...