解决ABA问题

解决ABA问题

一、ABA问题简述

在使用无锁CAS的时候,一个线程会先获取变量的值,比较的时候再获取内存中的值,如果一致就表示没有被其他线程修改过,然后就执行交换操作,但是如果一个线程修改了,然后另一个线程又修改为原来的值,这个时候一比较还是一样的。也就是说线程无法感知变量是否被修改过。

二、解决方法

使用AtomicStampedReference和版本号解决ABA问题,当前线程可以感知到值是否发生变化。

@Slf4j(topic = "c.atomicStampedReference")
public class AtmoicStampedRefInstance {
    
    private static AtomicStampedReference<String> ref = new AtomicStampedReference<>("A", 0);
    
    public static void main(String[] args) throws InterruptedException {
        log.info("main start...");
        String prev = ref.getReference();
        int stamp = ref.getStamp();
        log.info("{}", stamp);
        other();
        Thread.sleep(1000);
        //其他线程已经修改过了,CAS会失败,除了比较值还会比较版本号
        log.info("{}", stamp);
        log.info("change A->C {}", ref.compareAndSet(prev, "C", stamp, stamp + 1));
    }
    
    private static void other() throws InterruptedException {
        new Thread(() -> {
            int stamp = ref.getStamp();
            log.info("{}", stamp);
            log.info("change A->B {}", ref.compareAndSet(prev, "B", stamp, stamp + 1));
        }, "t1").start();
        new Thread(() -> {
            int stamp = ref.getStamp();
            log.info("{}", stamp);
            log.info("change B->A {}", ref.compareAndSet(prev, "A", stamp, stamp + 1));
        }, "t2").start();
    }
}

三、源码解析

/**
 * 维护了一个对象引用以及一个整型变量stamp
 */
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 V getReference() {
        return pair.reference;
    }
    
    /**
     * 返回stamp的当前值
     */
    public int getStamp() {
        return pair.stamp;
    }
    
    /**
     * 如果当前引用与期望的引用相同并且当前的stamp与期望的stamp相同,更新引用与stamp
     * @param expectedReference 期望的引用值
     * @param newReference 引用的新值
     * @param expectedStamp 期望的stamp值
     * @param newStamp 新的stamp值
     * @return 成功返回true
     */
    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)));
    }
    
    //...
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

快乐江小鱼

知识创造财富,期待您的慷慨解囊

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值