JUC并发编程学习笔记(十九)原子引用

发布时间 2023-11-14 19:25:18作者: 高同学,你好

原子引用

带版本号的原子操作!

解决ABA问题,引入原子引用(乐观锁思想)

AtomicStampedReference类解决ABA问题

package org.example.cas;

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

//使用原子引用解决ABA问题
public class ABADemo {
    public static void main(String[] args) {
        //默认值 默认版本号(时间戳)
        //如果泛型是一个包装类,注意对象引用的问题
        //正常在业务中里面比较并交换的一般是一个个对象如User这种
        AtomicStampedReference<Integer> atomic = new AtomicStampedReference<>(1,1);
        new Thread(()->{
            int stamp = atomic.getStamp();//获得版本号
            System.out.println("a1版本号=》"+stamp);
            try {
                TimeUnit.SECONDS.sleep(2);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            //atomic.compareAndSet 期待的值 更新的值 期待的版本号 更新的版本号
            //当期待的值和期待的版本号都满足期待时,就更新值和版本号
            System.out.println(atomic.compareAndSet(1, 2, stamp, stamp + 1));
            System.out.println("a2版本号=》"+atomic.getStamp());
            System.out.println(atomic.compareAndSet(2, 1, atomic.getStamp(), atomic.getStamp() + 1));
            System.out.println("a3版本号=》"+atomic.getStamp());
        },"A").start();
        //与乐观锁原理相同
        new Thread(()->{
            int stamp = atomic.getStamp();//获得版本号
            System.out.println("b1版本号=》"+stamp);
            try {
                TimeUnit.SECONDS.sleep(2);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println(atomic.compareAndSet(1, 2, stamp, stamp + 1));
            System.out.println("b2版本号=》"+atomic.getStamp());
        },"B").start();
    }
}

所有相同类型的包装类对象之间值的比较全部使用equals方法比较

Integer使用了对象缓存机制,默认范围是-128至127,推荐使用静态工厂方法valueOf获取对象实例,而不是new,因为valueOf使用缓存,而new一定会创建新的对象分配新的内存空间;

说明:对于Integer var = ?在-128至127之间的赋值,Integer对象实在IntegerCache.cache产生,会复用已有对象,这个区间的Integer值可以直接使用==进行判断,但是这个区间之外的所有数据,都会在堆上产生,并不会复用已有对象,这是一个大坑,推荐使用equals方法进行判断。