1. Cas通過當前值與輸入的期望值expert比較,判定是否改變值從而實現併發安全控制的方式。它可以不用加Synchromized實現線程安全問題。
缺點:
* 1. 循環時間長,開銷大(CPU負荷)
* 2. 只能保證單個共享變量的原子操作
* 3. ABA問題,通過版本問題解決
ABA問題,線程1執行 A-> D, 線程2執行 A->B->A,當線程2先執行完畢A又變爲A,線程1的期望值expect和當前值 A是一致的,仍然可以滿足執行條件,而這和原本設計目的相悖。解決辦法:加版本號
2. 代碼示例
import org.junit.Test;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.atomic.AtomicStampedReference;
/**
* @authod: pp_lan on 2020/4/2.
* CAS樣例
* 缺點:
* 1. 循環時間長,開銷大(CPU負荷)
* 2. 只能保證單個共享變量的原子操作
* 3. ABA問題,通過版本問題解決
*/
public class CasDemo {
@Test
/**
* Cas樣例
*/
public void test01() throws InterruptedException {
ExecutorService executor = Executors.newFixedThreadPool(2);
AtomicInteger atomicInteger = new AtomicInteger(5);
CountDownLatch latch = new CountDownLatch(2);
//compareAndSet,expert期望值及輸入值,如果值被修改了就和expert不一致,無法修改
executor.submit(() -> {
try {
TimeUnit.MILLISECONDS.sleep((long)(Math.random()*200));
} catch (InterruptedException e) {
e.printStackTrace();
}
boolean b = atomicInteger.compareAndSet(5, 2019);
System.out.println("2019\t" + b);
latch.countDown();
});
executor.submit(() -> {
try {
TimeUnit.MILLISECONDS.sleep((long)(Math.random()*200));
} catch (InterruptedException e) {
e.printStackTrace();
}
boolean b = atomicInteger.compareAndSet(5, 1024);
System.out.println("1024更新狀態\t" + b);
latch.countDown();
});
//閥門等待線程執行完畢
latch.await();
System.out.println("value============= " + atomicInteger.get());
}
@Test
/**
* 原子引用
*/
public void test02() {
AtomicReference<User> atomicReference = new AtomicReference<>();
User user1 = new User("zhangsan", 23);
User user2 = new User("lisi", 24);
atomicReference.set(user1);
System.out.println(atomicReference.compareAndSet(user1, user2) + "\t" + atomicReference.get());
//無法更新
System.out.println(atomicReference.compareAndSet(user1, user1) + "\t" + atomicReference.get());
}
/**
* Junit中使用sleep會不運行,此處改爲main方法測試號
* @param args
*/
public static void main(String[] args) throws InterruptedException {
System.out.println("================= ABA問題復現 =====================");
CountDownLatch latch1 = new CountDownLatch(2);
AtomicReference<Integer> atomicReference = new AtomicReference<>(100);
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + "\t執行");
atomicReference.compareAndSet(100, 101);
atomicReference.compareAndSet(101, 100);
latch1.countDown();
}, "thread1").start();
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + "\t執行");
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
atomicReference.compareAndSet(100, 2019);
System.out.println("最新value:\t" + atomicReference.get());
latch1.countDown();
}, "thread2").start();
latch1.await();
System.out.println("================= ABA問題復現結束 =====================");
System.out.println("================= ABA問題解決開始 =====================");
CountDownLatch latch2 = new CountDownLatch(2);
AtomicStampedReference<Integer> atomicStampedReference = new AtomicStampedReference<>(100, 1);
new Thread(() -> {
int stamp = atomicStampedReference.getStamp();
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
atomicStampedReference.compareAndSet(100, 101,
atomicStampedReference.getStamp(), atomicStampedReference.getStamp() + 1);
atomicStampedReference.compareAndSet(101, 102,
atomicStampedReference.getStamp(), atomicStampedReference.getStamp() + 1);
atomicStampedReference.compareAndSet(102, 100,
atomicStampedReference.getStamp(), atomicStampedReference.getStamp() + 1);
latch2.countDown();
}, "thread1").start();
new Thread(() -> {
int stamp = atomicStampedReference.getStamp();
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("使用版本號\t" + stamp);
System.out.println("實際版本號\t" + atomicStampedReference.getStamp());
boolean b = atomicStampedReference.compareAndSet(100, 2020, stamp, stamp + 1);
System.out.println("舊版本號執行結果\t" + b);
latch2.countDown();
}, "thread2").start();
latch2.await();
System.out.println("執行結果:\t" + atomicStampedReference.getReference());
System.out.println("================= ABA問題解決結束 =====================");
}
class User {
private String username;
private int age;
public User(String username, int age) {
this.username = username;
this.age = age;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
@Override
public String toString() {
return "User{" +
"username='" + username + '\'' +
", age=" + age +
'}';
}
}
}