原子引用
带版本号的原子操作!
解决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方法进行判断。
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |