ABA问题分析与解决

ABA问题分析

产生原因

CAS会导致ABA问题

CAS算法实现一个重要的前提是需要取出内存中某时刻的数据并在当下时刻比较并替换,那么在这个时间差类会导致数据的变化.

比如说一个线程one从内存位置V中取出A,线程Two也执行,将A–>B–>A,这时one进行CAS操作发现内存仍是A,然后one操作成功.

one操作成功,但是在这个过程中线程two可能操作了其它数据,产生问题.

package com.ygy.juc;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;

/**
 * 产生ABA问题
 */
public class ABADemo {

    static AtomicReference<Integer> atomicReference = new AtomicReference<>(100);

    public static void main(String[] args) {
        new Thread(()->{
            atomicReference.compareAndSet(100,101);
            atomicReference.compareAndSet(101,100);
        },"t1").start();

        new Thread(()->{
            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(atomicReference.compareAndSet(100,2019)+"\t"+atomicReference.get());
        }).start();
    }
}
	

解决办法

如果线程one操作不被中断,那么问题就解决了

  • 将类变成原子类
  • 操作过程添加版本号

原子引用

在juc包中有默认的原子类型原子操作类,如果不满足需求,可以自己创造原子类型,用原子引用,AtomicReference,可以包装类,达到操作不被中断的目的

package com.ygy.juc;

import java.util.concurrent.atomic.AtomicReference;

/**
 * @Description: TODO(原子引用)
 * @author yangguangyuan
 * @date 2019年6月14日
 *
 */
public class AtomicReferenceDemo {
	public static void main(String[] args) {
		User z3 = new User("zhangsan", 23);
		User li4 = new User("lisi", 23);

		AtomicReference<Object> atomicReference = new AtomicReference<>();
		atomicReference.set(z3);

		System.out.println(atomicReference.compareAndSet(z3, li4) + "\t"
				+ atomicReference.get().toString());
		System.out.println(atomicReference.compareAndSet(z3, li4) + "\t"
				+ atomicReference.get().toString());
	}
}

class User {
	String username;
	int age;

	public User(String username, int age) {
		this.username = username;
		this.age = age;
	}

	@Override
	public String toString() {
		return "User [username=" + username + ", age=" + age + "]";
	}
}

问题

虽然已经变成原子类,但是还是没有解决ABA问题

时间戳原子引用

原子引用+新增机制(修改版本号,类似时间戳)

package com.ygy.juc;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.atomic.AtomicStampedReference;

/**
 * 解决ABA问题
 */
public class ABADemo2 {

    static AtomicStampedReference<Integer> reference = new AtomicStampedReference<>(100,1);
    public static void main(String[] args) {
        new Thread(()->{
            int stamp = reference.getStamp();
            System.out.println("第1次版本号:"+stamp);
            try {
                TimeUnit.SECONDS.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            reference.compareAndSet(100,101,reference.getStamp(),reference.getStamp()+1);
            System.out.println("第2次版本号:"+reference.getStamp());
            reference.compareAndSet(101,100,reference.getStamp(),reference.getStamp()+1);
            System.out.println("第3次版本号:"+reference.getStamp());
        },"t3").start();

        new Thread(()->{
            int stamp = reference.getStamp();
            System.out.println("第1次版本号:"+stamp);
            try {
                TimeUnit.SECONDS.sleep(3);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            boolean result = reference.compareAndSet(100,2019,stamp,stamp+1);
            System.out.println(Thread.currentThread().getName()+"修改是否成功:"+result+"\t当前最新实际版本号:"+reference.getStamp());
            System.out.println(Thread.currentThread().getName()+"\t当前实际最新值:"+reference.getReference());
        },"t4").start();

    }
}

  • 0
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值