aba会导致问题_JUC多线程与高并发面试题——ABA问题与原子引用

一、ABA问题怎么产生的

CAS会导致“ABA问题”。

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

比如:一个线程ONE从内存位置V中取出A,这时候线程TWO也从内存中取出A,并且线程TWO进行了一些操作将值变成了B,然后线程TWO又将V位置的数据变成A,此时线程ONE进行CAS操作,发现内存中仍然是A,然后线程ONE操作成功。

尽管线程ONE的CAS操作成功,但是并不代表这个过程是没有问题的。

二、原子引用

package com.yuxx.juc;

import lombok.AllArgsConstructor;

import lombok.Getter;

import lombok.ToString;

import java.util.concurrent.atomic.AtomicReference;

@Getter

@ToString

@AllArgsConstructor

class User{

String userName;

int age;

}

public class AtomicReferenceDemo {

public static void main(String[] args) {

User z3 = new User("z3",22);

User l4 = new User("l4",25);

AtomicReference atomicReference = new AtomicReference();

atomicReference.set(z3);

System.out.println(atomicReference.compareAndSet(z3,l4) + "\t" + atomicReference.get().toString());//true User(userName=l4, age=25)

System.out.println(atomicReference.compareAndSet(z3,l4) + "\t" + atomicReference.get().toString());//falseUser(userName=l4, age=25)

}

}

打印结果:

三、ABA问题解决——时间戳原子引用

时间戳原子引用:AtomicStampedReference

package com.yuxx.juc;

import java.util.concurrent.TimeUnit;

import java.util.concurrent.atomic.AtomicStampedReference;

public class ABADemo { //ABA问题的解决 AtomicStampedReference

static AtomicStampedReference atomicStampedReference = new AtomicStampedReference<>(100,1);

public static void main(String[] args) {

new Thread(() -> {

System.out.println(Thread.currentThread().getName() + "\t第1次版本号:" + atomicStampedReference.getStamp());

//暂停1秒钟T3线程

try{ TimeUnit.SECONDS.sleep(1); }catch(InterruptedException e){ e.printStackTrace(); }

atomicStampedReference.compareAndSet(100,101,

atomicStampedReference.getStamp(),atomicStampedReference.getStamp()+1);

System.out.println(Thread.currentThread().getName() + "\t第2次版本号:" + atomicStampedReference.getStamp());

atomicStampedReference.compareAndSet(101,100,

atomicStampedReference.getStamp(),atomicStampedReference.getStamp()+1);

System.out.println(Thread.currentThread().getName() + "\t第3次版本号:" + atomicStampedReference.getStamp());

},"T3").start();

new Thread(() -> {

int stamp = atomicStampedReference.getStamp();

System.out.println(Thread.currentThread().getName() + "\t第1次版本号:" + stamp);

//暂停3秒钟T4线程,保证上面的T3线程完成了一次ABA操作

try{ TimeUnit.SECONDS.sleep(3); }catch(InterruptedException e){ e.printStackTrace(); }

boolean result = atomicStampedReference.compareAndSet(100, 2019, stamp, stamp + 1);

System.out.println(Thread.currentThread().getName() + "\t修改成功否:" + result + "\t当前最新实际版本号:" + atomicStampedReference.getStamp());

System.out.println(Thread.currentThread().getName() + "\t当前实际最新值:" + atomicStampedReference.getReference());

},"T4").start();

}

}

打印结果:

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值