返回顶部
首页 > 资讯 > 后端开发 > Python >Java中的CAS和ABA问题说明
  • 481
分享到

Java中的CAS和ABA问题说明

2024-04-02 19:04:59 481人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

目录1.CAS1)CAS概念2)CAS产生的影响(无锁执行)3)Automic并发类CAS原理代码分析4)CAS导致的ABA问题1.CAS 1)CAS概念 CAS时Compare A

1.CAS

1)CAS概念

CAS时Compare And Swap缩写,即比较与交换是用于实现多线程同步的原子指令,它将内存位置的内容与给定值相比较,相同则修改内存位置的值为新值,而整个操作是调用的UnSafe的compareAndSwapObject、compareAndSwapint或者compareAndSwapLong完成的,而这些方法都是native修饰的本地方法,是一种系统原语系统支持的操作。

2)CAS产生的影响(无锁执行)

CAS是一种无锁对象的原子操作,锁分为乐观锁和悲观锁,乐观派抱着几乎不会发生修改同一资源的状态,任意操作同意对象资源,如果遇到修改同一资源的情况,资源不会修改成功,能够保证资源的安全,而悲观派会认为同一资源被错误修改后会造成不可挽回的局面,故自能有一个线程修改资源,这样总会对系统性能产生一定的影响,拖慢自行速度,CAS即无锁执行者,被CAS修饰过的资源可以同时被多个线程修改依然能保证系统安全,无锁不需要等待提高系统性能,jdk提供的CAS原理实现的并发类Automic系列运用及其原理介绍。

3)Automic并发类CAS原理代码分析

首先介绍java的指针操作类UnSafe,Unsafe类是在sun.misc包下,不属于Java标准。但是很多Java的基础类库,包括一些被广泛使用的高性能开发库都是基于Unsafe类开发的,因为UnSafe使Java像C语言一样使其拥有操作内存指针的能力,因为操作内存指针容易出错,故起名UnSafe不安全的类,因此Java官方并不建议使用的,但CAS原理就是UnSafe类中的compareAndSwapObject、compareAndSwapInt和compareAndSwapLong方法实现的,该方法需传入四个参数:第一个参数代表给定的对象,第二个参数代表给定对象再内存中的偏移量,第三个参数标识对象的期望值,第四个参数标识要修改的值,并发保重的Automic系列的原子操作类都是使用UnSafe类实现的。

UnSafe源码如下:


public final native boolean compareAndSwapObject(Object var1, long var2, Object var3, Object var4);
    public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);
    public final native boolean compareAndSwapLong(Object var1, long var2, long var4, long var6);

举例AtomicInteger源码实现原理:

AutomicInteger中的getAndSet实现原理解析:

    
    public final int getAndSet(int newValue) {
        return unsafe.getAndSetInt(this, valueOffset, newValue);
    }
package java.util.concurrent.atomic;
import java.util.function.IntUnaryOperator;
import java.util.function.IntBinaryOperator;
import sun.misc.Unsafe;
 
public class AtomicInteger extends Number implements java.io.Serializable {
    private static final long serialVersionUID = 6214790243416807050L;
 
    // 获取UnSafe对象实例
    private static final Unsafe unsafe = Unsafe.getUnsafe();
    //对象在内存中的偏移量
    private static final long valueOffset;
    
    //初始化valueOffset
    static {
        try {
            valueOffset = unsafe.objectFieldOffset
                (AtomicInteger.class.getDeclaredField("value"));
        } catch (Exception ex) { throw new Error(ex); }
    }
    
    //对象属性值
    private volatile int value;
 
    public AtomicInteger(int initialValue) {
        value = initialValue;
    }
 
    public AtomicInteger() {
    }
 
    
    public final int getAndSet(int newValue) {
        return unsafe.getAndSetInt(this, valueOffset, newValue);
    }
 
    //调用UnSafe的compareAndSwapInt方法保证CAS
    public final boolean compareAndSet(int expect, int update) {
        return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
    }
 
    //调用UnSafe的compareAndSwapInt方法保证CAS
    public final boolean weakCompareAndSet(int expect, int update) {
        return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
    }
 
    //调用UnSafe的getAndAddInt再调用UnSafe的getAndSetInt方法保证CAS
    public final int getAndIncrement() {
        return unsafe.getAndAddInt(this, valueOffset, 1);
    }
 
    
    public final int getAndDecrement() {
        return unsafe.getAndAddInt(this, valueOffset, -1);
    }
    .........
}

4)CAS导致的ABA问题

操作对象,获取对象后,执行CAS操作前,被其他线程修改后,且又修改为原来的对象值,导致CAS忽略其他线程的修改,成功执行CAS对象修改,这种情况就叫做ABA问题。

下图所示:

解决办法:

AtomicStampedReference类提供了解决办法,在对象之中又添加了stamp时间戳属性避免其他线程修改了多次并变回修改前的value值,但对比stamp不同便可知道对象是被修改过的,只有提供属性值和stamp时间戳相等才能成功执行CAS修改操作,里面包裹了一个键值对对象AtomicStampedReference.Pair<V> pair类型,pair中值为属性值,value为stamp时间戳,在执行CAS操作时需要提供原值的value和时间戳都相等的情况才能成功执行CAS操作。

AtomicMarkableReference类提供了解决办法,在对象之中又添加了stamp时间戳属性避免其他线程修改了多次并变回修改前的value值,但对比stamp不同便可知道对象是被修改过的,只有提供属性值和boolean类型的mark标记相等才能成功执行CAS修改操作,里面包裹了一个键值对对象AtomicMarkableReference.Pair<V> pair类型,pair中值为属性值,value为mark是否被修改的标记,在执行CAS操作时需要提供原值的value和mark标记都相等的情况才能成功执行CAS操作。

本文只介绍AtomicStampedReference类的源码分析,AtomicMarkableReference类同AtomicStampedReference类原理一样,

源码如下:

package java.util.concurrent.atomic;
public class AtomicStampedReference<V> {
   
    private static class Pair<T> {
        final T reference;
        final int stamp;
        private Pair(T reference, int stamp) {
            this.reference = reference;
            this.stamp = stamp;
        }
        static <T> Pair<T> of(T reference, int stamp) {
            return new Pair<T>(reference, stamp);
        }
    }
 
    private volatile Pair<V> pair;
 
    
    public AtomicStampedReference(V initialRef, int initialStamp) {
        pair = Pair.of(initialRef, initialStamp);
    }
    
    
    public boolean compareAndSet(V   expectedReference,
                                 V   newReference,
                                 int expectedStamp,
                                 int newStamp) {
        Pair<V> current = pair;  //赋值当前对象
        return
            expectedReference == current.reference &&
            expectedStamp == current.stamp &&
            ((newReference == current.reference &&
              newStamp == current.stamp) ||
             casPair(current, Pair.of(newReference, newStamp)));
    }
 
    
    // Unsafe mechanics
 
    private static final sun.misc.Unsafe UNSAFE = sun.misc.Unsafe.getUnsafe();
    private static final long pairOffset =
        objectFieldOffset(UNSAFE, "pair", AtomicStampedReference.class);
 
    private boolean casPair(Pair<V> cmp, Pair<V> val) {
        return UNSAFE.compareAndSwapObject(this, pairOffset, cmp, val);
    }
 
    static long objectFieldOffset(sun.misc.Unsafe UNSAFE,
                                  String field, Class<?> klazz) {
        try {
            return UNSAFE.objectFieldOffset(klazz.getDeclaredField(field));
        } catch (NoSuchFieldException e) {
            // Convert Exception to corresponding Error
            NoSuchFieldError error = new NoSuchFieldError(field);
            error.initCause(e);
            throw error;
        }
    }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

--结束END--

本文标题: Java中的CAS和ABA问题说明

本文链接: https://lsjlt.com/news/148139.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java中的CAS和ABA问题说明
    目录1.CAS1)CAS概念2)CAS产生的影响(无锁执行)3)Automic并发类CAS原理代码分析4)CAS导致的ABA问题1.CAS 1)CAS概念 CAS时Compare A...
    99+
    2024-04-02
  • 详解java 中的CAS与ABA
    目录1. 独占锁: 1.1 乐观锁的操作 2. 乐观锁: 2.1 CAS操作 3. 原子变量类 4. CAS的缺陷 1. 独占锁: 属于悲观锁,有共享资源,需要加锁时,会以独占锁的...
    99+
    2024-04-02
  • 解析Java多线程之常见锁策略与CAS中的ABA问题
    目录1.常见的锁策略1.1乐观锁与悲观锁1.2读写锁与普通互斥锁1.3重量级锁与轻量级锁1.4挂起等待锁与自旋锁1.5公平锁与非公平锁1.6可重入锁与不可重入锁1.7死锁问题1.7....
    99+
    2024-04-02
  • Java多线程之常见锁策略与CAS中的ABA问题怎么解决
    本文小编为大家详细介绍“Java多线程之常见锁策略与CAS中的ABA问题怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java多线程之常见锁策略与CAS中的ABA问题怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一...
    99+
    2023-06-30
  • Java中多线程的ABA场景问题分析
    目录前言一、异步场景常用工具二、CAS 与 ABA 问题三、用 JUC 工具处理 ABA 问题总结前言 本文是笔者在日常开发过程中遇到的对 CAS 、 ABA 问题以及 JUC(ja...
    99+
    2022-12-29
    Java 多线程ABA Java 多线程
  • Java并发中的ABA问题学习与解决方案
    目录1.简介2.Compare and swap3. ABA问题3.1 ABA问题的实际场景:账户余额修改3.2 账户余额修改时产生的问题4.银行取款问题代码演示5.值类型与引用类型...
    99+
    2024-04-02
  • Java非法字符:‘\ufeff‘问题及说明
    目录Java非法字符: ‘\ufeff‘解决办法项目编译报错:java 非法字符 \ufeffcreate utf-8 files:with no bom项目...
    99+
    2023-02-25
    Java非法字符 Java ‘\ufeff‘ 非法字符 ‘\ufeff‘
  • C语言中的多行输入问题及说明
    目录一、未明确具体输入行数1、~scanf方式2、EOF3、根据输入元素的个数二、已知具体输入行数总结牛客网和LeetCode的竞赛题中经常会遇到多行输入,以前不熟悉这种写程序的方式...
    99+
    2023-02-05
    C语言多行输入 多行输入 多行输入问题
  • 迭代器的应用以及说明的问题
    一般使用for循环的顺序:   先创建一个迭代器:列表,并且列表内已存在数据   接着使用 for循环遍历列表   但这样存在一个问题,假如列表内的数据量很大,这样还未使用for循环便已经被列表占用很大的资源 这时我们就可以将这个迭代器做...
    99+
    2023-01-30
    迭代
  • java中1+1d/5和1+1/5的区别说明
    目录java 1+1d/5和1+1/5的区别主要区别就是数据类型不同计算1+1/2!+1/3!+...+1/20!之和java 1+1d/5和1+1/5的区别 主要区别就是数据类型不...
    99+
    2024-04-02
  • Java中import导入的用法说明
    目录import导入的用法导入方式java中import作用packageimportimport的两种导入声明static import静态导入按需导入机制附加 impo...
    99+
    2024-04-02
  • Java Scanner的使用和hasNextXXX()的用法说明
    目录输入输出输出输入使用Scanner读取使用Scanner循环读取N个数字/字符串关于Scanner中nextxxx()须注意的一点输入输出 输出 基本语法 System.ou...
    99+
    2024-04-02
  • mysql中binlog和redo的说明和对比
    本篇内容介绍了“mysql中binlog和redo的说明和对比”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成...
    99+
    2024-04-02
  • Java 关键字break和continue的使用说明
    break和continue的说明 break   循环结构,一旦执行,就结束(或跳出)当前循环结构,此关键字的后面,不能声明语句continue 循环结构,一旦...
    99+
    2023-03-22
    Java 关键字break和continue使用 Java 关键字break和continue
  • golang中json和struct的使用说明
    1、返回json响应结果 在struct的字段后面加入json:"key"可以进行json格式输出,其中key为json的键名 type SuccessResponse stru...
    99+
    2024-04-02
  • Pytorch 中net.train 和 net.eval的使用说明
    在训练模型时会在前面加上: model.train() 在测试模型时在前面使用: model.eval() 同时发现,如果不写这两个程序也可以运行,这是因为这两个方法是针...
    99+
    2024-04-02
  • Scala中Array和List的区别说明
    目录Scala Array和List的区别Scala快排List和Array数组效率实测Scala Array和List的区别 Difference between Array an...
    99+
    2024-04-02
  • vue中data和data()的区别说明
    目录data和data()的区别Vue实例中data属性组件化的项目中使用详解vue.js中的data文档之一文档之二文档之三文档之四data和data()的区别 Vue实例中dat...
    99+
    2024-04-02
  • Collection中的size()和isEmpty()区别说明
    目录Collection中的size()和isEmpty()区别说明Collection集合类介绍与实验list.size()和list.isEmpty()区别和效率及Collect...
    99+
    2024-04-02
  • vue中的addEventListener和removeEventListener用法说明
    目录addEventListener和removeEventListener用法说明1、添加监听事件(addEventListener)2、移出监听事件(removeEventLis...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作