Java并发_6 CAS

CAS

什么是CAS

CompareAndSwap,是一种思想和算法,由CPU指令保证原子性。

我认为V的值应该是A,如果是的话那我就把它改成B,如果不是A(说明被别人修改过了),那我就不修改了,避免多人同时修改导致出错。

CAS有三个操作数︰内存值V、预期值A、要修改的值B,当且仅当预期值A和内存值V相同时,才将内存值修改为B,否则什么都不做。最后返回bool值。

算法本质

其实是比较简单的

private volatile int value;public synchronized boolean compareAndSwap(int expect,int newValue){int oldValue = value;    if (oldValue == expect){value = newValue;        return true;    }    return false;}

应用场景

乐观锁

数据库在修改的时候可以使用版本号的方式去修改

并发容器

比如ConcurrentHashMap中的put方法就使用到了casTabAt,可以看到名字中就有cas,而它调用了unsafe的native方法compareAndSwapObject。

源码:分析Java如何利用CAS实现原子操作

以AtomicInteger为例:

  1. AtomicInteger加载Unsafe工具,用来直接操作内存数据

  2. 用Unsafe来实现底层操作

  3. 用volatile修饰value字段,保证可见性

  4. getAndAddInt方法分析

static {try {valueOffset = unsafe.objectFieldOffset            (AtomicInteger.class.getDeclaredField("value"));    } catch (Exception ex) { throw new Error(ex); }}
public final int getAndAddInt(Object var1, long var2, int var4) {int var5;    do {var5 = this.getIntVolatile(var1, var2);    } while(!this.compareAndSwapInt(var1, var2, var5, var5   var4));    return var5;}

Unsafe

Unsafe是CAS的核心类。Java无法直接访问底层操作系统,而是通过本地( native )方法来访问。不过尽管如此,JVM还是开了一个后门,JDK中有一个类Unsafe,它提供了硬件级别的原子操作。

valueOffset表示的是变量值在内存中的偏移地址,因为Unsafe就是根据内存偏移地址获取数据的原值的,这样我们就能通过unsafe来实现CAS了。

UNSAFE_ENTRY(jboolean,Unsafe_CompareAndSwapInt(JNIEnv *env, jobject unsafe,jobject obj, jlong offset, jint e, jint x))UnsafeWrapper("Unsafe_CompareAndSwapInt");oop p = JNIHandlesresolve(obj);//通过偏移量得到内存地址jint* addr = (jint *) index_oop_from_field_offset_long(p,offset);//通过内存地址就可以执行CAS了,x为更新的值,e为原值return (jint)(Atomic::cmpxchg(x, addr, e))== e;UNSAFE_END

缺点

ABA问题

自旋时间太长,比如unsafe中使用了do-while循环。这会导致消耗cpu资源。

CAS的深度拓展

CAS的ABA问题

如果在CAS执行间有线程已经实施了一定的操作,但是经过几次修改之后的数据和原来的一样,那么CAS就可能无法判断是否改变过,如果这样的情况不允许发生那么可以使用版本号,或者bool值来解决。

CAS的底层原理

CAS的本质和synchronized相同(LOCK CMPXCHG),汇编中都是采用了指令集的LOCK前缀,即在所对应的指令操作期间使此指令的目标操作数指定的存储区域的北桥信号(每个核心都经过都需要看)锁定,以得到保护。

CMPXCHG:比较交换指令,第一操作数先和AL/AX/EAX比较,如果相等ZF置1,第二操作数赋给第一操作数,否则ZF清0,第一操作数赋给AL/AX/EAX。多处理器安全,在80486及以上CPU中支持。

关于volatile

volatile使用的是LOCK; ADDL $0,0(%%rsp)往某个寄存器中加0,其实就是空指令,主要是使用LOCK来达到内存可见性。

lock前缀的作用

  • 将当前处理器缓存行的数据会写回到系统内存。

  • 这个写回内存的操作会引起在其他 CPU 里缓存了该内存地址的数据无效。

volatile伪共享

解决方便是将高速缓存剩余的字节填充填满(pad),确保不发生多个字段被挤入一个高速缓存区。


无限扩展等等

来源:https://www.icode9.com/content-1-863101.html

(0)

相关推荐

  • 浅谈 Java 并发下的乐观锁

    引子 各位少侠大家好!今天我们来聊聊 Java 并发下的乐观锁. 在聊乐观锁之前,先给大家复习一个概念:原子操作: 什么是原子操作呢? 我们知道,原子(atom)指化学反应不可再分的基本微粒.在 Ja ...

  • Java高并发9-CAS操作与Unsafe类详解

    一.复习 计算机内存模型,synchronized和volatile关键字简介 二.两者对比 sychronized和volatile都解决了内存可见性问题 不同点: (1)前者是独占锁,并且存在者上 ...

  • Java高并发11-伪共享,getUnsafe源码解析并利用反射获取Unsafe实例

    一.复习 public native long getLongvolatile(Object obj,long offset) public native long putLongvolatile(O ...

  • 思维导图整理Java并发基础知识

    话不多说,先上图. 1.基本概念 欲说线程,必先说进程. 进程 :进程是代码在数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位. 线程 :线程是进程的一个执行路径,一个进程中至少有一个线程 ...

  • 【JAVA并发第三篇】线程间通信

    线程间的通信 JVM在运行时会将自己管理的内存区域,划分为不同的数据区,称为运行时数据区.每个线程都有自己私有的内存空间,如下图示: Java线程按照自己虚拟机栈中的方法代码一步一步的执行下去,在这一 ...

  • 【JAVA并发第四篇】线程安全

    【JAVA并发第四篇】线程安全

  • Java并发编程之内置锁(synchronized)

    简介 synchronized在JDK5.0的早期版本中是重量级锁,效率很低,但从JDK6.0开始,JDK在关键字synchronized上做了大量的优化,如偏向锁.轻量级锁等,使它的效率有了很大的提 ...

  • Java并发编程之线程的创建

    简介 线程是基本的调度单位,它被包含在进程之中,是进程中的实际运作单位,它本身是不会独立存在.一个进程至少有一个线程,进程中的多个线程共享进程的资源. Java中创建线程的方式有多种如继承Thread ...

  • Java并发编程实战(5)- 线程生命周期

    在这篇文章中,我们来聊一下线程的生命周期. 目录 概述 操作系统中的线程生命周期 Java中的线程生命周期 Java线程状态转换 运行状态和阻塞状态之间的转换 运行状态和无时限等待状态的切换 运行状态 ...

  • 100道Java并发和多线程基础面试题大集合(含解答),这波面试稳了~

    # 前言 这篇文章主要是对多线程的问题进行总结的,因此罗列了100个多线程的问题. 这些多线程的问题来源于各大网站,可能有些问题网上有.可能有些问题对应的答案也有.也可能有些各位网友也都看过,但是本文 ...

  • Java并发多线程编程——Volatile原理与使用

    优质文章,第一时间送达 76套java从入门到精通实战课程分享 一.volitile的理解 Volatile称之为轻量级锁,被volatile修饰的变量,在线程之间是可见的. 可见即一个线程修改了这个 ...

  • Java并发编程实战(4)- 死锁

    概述 在上一篇文章中,我们讨论了如何使用一个互斥锁去保护多个资源,以银行账户转账为例,当时给出的解决方法是基于Class对象创建互斥锁. 这样虽然解决了同步的问题,但是能在现实中使用吗?答案是不可以, ...