原子引用
带版本号的原子操作!
解决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方法进行判断。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/199831.html