当前位置: 首页 > news >正文

上海哪家做网站关键词排名北京企业网站seo平台

上海哪家做网站关键词排名,北京企业网站seo平台,廉江网站建设公司,网站建设业务前景一、死磕Java——CAS 前面我们说到volatile不保证原子性,解决办法就是使用AtomicInteger代替int,但是为什么使用AtomicInteger就可以保证了原子性了,是因为AtomicInteger实现的就是CAS思想和Unsafe的支持。 1.1.CAS是什么 AtomicInteger atom…

一、死磕Java——CAS

前面我们说到volatile不保证原子性,解决办法就是使用AtomicInteger代替int,但是为什么使用AtomicInteger就可以保证了原子性了,是因为AtomicInteger实现的就是CAS思想Unsafe的支持。

1.1.CAS是什么

AtomicInteger atomicInteger = new AtomicInteger(5);
atomicInteger.compareAndSet(5, 10);
复制代码

CAS:即比较和交换(compareAndSet),CAS的思想比较简单就是三个值:当前内存值V,旧的预期值A,和要更新的值B,当且仅当内存值V等于预期值A,才将内存值修改为B,并返回true,否则什么都不做,返回false。下面就以atomicInteger.getAndIncrement();分析一下AtomicInteger使用的CAS思想。

1.2.CAS的原理

使用AtomicInteger

public static void main(String[] args) {AtomicInteger atomicInteger = new AtomicInteger(5);atomicInteger.compareAndSet(5, 10);atomicInteger.getAndIncrement();
}
复制代码

compareAndSet方法

public final int getAndIncrement() {return unsafe.getAndAddInt(this, valueOffset, 1);
}
复制代码

unsafe.getAndAddInt(this, valueOffset, 1)说明:

  • this:代表当前对象,这里就是外部newatomicInteger对象;
  • valueOffset:代表当前对象的内存地址。
  • 1:就是加1。
private static final Unsafe unsafe = Unsafe.getUnsafe();
private static final long valueOffset;static {try {valueOffset = unsafe.objectFieldOffset(AtomicInteger.class.getDeclaredField("value"));} catch (Exception ex) { throw new Error(ex); }
}private volatile int value;
复制代码

说明:

上述代码就是根据对象的内存地址获取当前内存的值,注意的是private volatile int value;添加了volatile关键字,所以,保证了可见性。

unsafe.getAndAddInt方法

public final int getAndAddInt(Object var1, long var2, int var4) {int var5;do {var5 = this.getIntVolatile(var1, var2);} while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));return var5;
}
复制代码

说明:

  • unsafe类是CAS的核心类,由于Java无法直接操作底层系统,只能通过native修饰的本地方法操作,基于unsafe类就可以直接操作内存中的数据。
  • getAndAddInt(Object var1, long var2, int var4)中:Object var1:当前对象,long var2:当前对象的内存地址,int var4:需要更新的值,这里就是1。
  • var5 = this.getIntVolatile(var1, var2);:就是取到当前对象的内存值;
  • 假如现在存在两个线程,跑在不同的cpu上,内存中atomicInteger的原始值为5,两个线程都拷贝一份到自己的工作内存中,
  • 线程A通过getIntVolatile(var1, var2)拿到value值5,这时线程A被挂起。
  • 线程B也通过getIntVolatile(var1, var2)方法获取到value值5,线程B没有被挂起,并执行compareAndSwapInt方法比较内存值也为5,成功修改内存值为10。
  • 这时线程A恢复,执行compareAndSwapInt方法比较,发现自己手里的值(5)和内存的值(10)不一致,说明该值已经被其它线程提前修改过了,那只能重新来一遍了。
  • 重新获取value值,因为变量valuevolatile修饰,所以其它线程对它的修改,线程A总是能够看到,线程A继续执行compareAndSwapInt进行比较替换,直到成功。

1.3.CAS的缺点

前面的代码分析中我们知道getAndAddInt方法有一个do-while循环,如果CAS失败,就会进行一直尝试比较,如果很长时间都不成功,就会增加CPU的开销。所以CAS的一个缺点就是循环时间长开销大,由于this表示的是当前对象,所以,存在另外一个缺点就是只能保证一个共享变量的原子操作。最重要的缺点就是ABA问题

1.3.1.什么是ABA问题

前面分析CAS思想的时候,我们知道一个线程会先获取Value的值,比较和交换的时候再获取内存的值和手里的value进行比较,说的是如果一致就表示没有被其他线程修改过,然后就执行自己的交换操作,但是,如果,一个线程修改了,然后另外还有一个线程又修改会原来的值,这个时候一比较还是一样的,这就是ABA问题。简单讲就是狸猫换太子。如果业务中不关心中间操作,只在乎开始和结尾是否一致就可,就不必要解决ABA 问题。

1.3.2.使用原子引用和版本机制解决ABA问题

java.util.concurrent.atomic包下存在一个AtomicReference类,就是原子引用,CAS比较的只是内存中的值,现在增加一个版本号,比较值的同时再比较版本后是否一致。使用AtomicStampedReference带时间戳的原子引用来解决ABA问题。

 public static void main(String[] args) {AtomicStampedReference<Integer> reference = new AtomicStampedReference<>(10, 1);new Thread(() -> {int stamp = reference.getStamp();System.out.println("T1拿到的第一次的版本号:" + stamp);// 先暂停1秒,等T2线程拿到相同的初始版本号try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}reference.compareAndSet(10, 101, reference.getStamp(), reference.getStamp() + 1);System.out.println("T1线程第一次操作后的版本号为:"  + reference.getStamp());reference.compareAndSet(101, 10, reference.getStamp(), reference.getStamp() + 1);System.out.println("T1线程第二次操作后的版本号为:"  + reference.getStamp());}, "T1").start();new Thread(() -> {int stamp = reference.getStamp();System.out.println("T2拿到的第一次的版本号:" + stamp);// 先暂停3秒,等T1线程有充分的时候做一次ABA操作try {TimeUnit.SECONDS.sleep(3);} catch (InterruptedException e) {e.printStackTrace();}boolean b = reference.compareAndSet(10, 2019, stamp, stamp + 1);System.out.println("当前内存中的最新值为:" + reference.getReference());System.out.println("T2线程在T1线程执行完ABA问题后在执行的结果为:" + b);}, "T2").start();}
复制代码
http://www.lbrq.cn/news/2570239.html

相关文章:

  • 浙江网站建设品牌升级汕头网站制作设计
  • 做公益做的好的的网站济南seo网络优化公司
  • 男的做直播哪个网站友情链接检测平台
  • 青岛网站建设ihuibest深圳正规seo
  • 微网站难做么合肥优化营商环境
  • wordpress导航栏的文件在哪太原seo自媒体
  • 自己电脑怎样做网站关键词快速排名seo怎么优化
  • 备案怎么关闭网站整合营销传播的方法包括
  • 成都市武侯区建设局门户网站链接购买
  • 平面网页设计培训宁波网站关键词优化代码
  • 传媒的域名做个什么网站上海seo优化服务公司
  • 湖南省住房和建设厅网站山西seo
  • 北京网站制作报价域名解析查询站长工具
  • 做网站要花钱吗seo引擎优化是做什么的
  • 网站个人建设亚马逊排名seo
  • 什么2007做视频网站服装品牌营销策划方案
  • 哪里有做网站app的泰州seo平台
  • 自己做衣服网站网络营销策划的目的
  • 做网站的系统功能需求产品推广文案范文
  • 网站建设设计设计成都纯手工seo
  • 淘宝网首页官网登录焦作seo推广
  • 那些网站可以接私活做seo网络推广企业
  • 免费个人简历制作网站怎么网络推广自己业务
  • 网页设计要多少钱电脑系统优化工具
  • 网站建设一条龙ue365企业文化标语
  • 肥西网站建设教育培训机构有哪些
  • 网站pv统计方法站长之家音效
  • 痘痘怎么去除效果好网站关键词优化排名外包
  • 移动端网站如何优化最新全国疫情消息
  • 政府门户网站升级建设方案广州 竞价托管
  • PyTorch 中 Tensor 统计学函数及相关概念
  • 下载一个JeecgBoot-master项目 导入idea需要什么操作启动项目
  • Go语言中的盲点:竞态检测和互斥锁的错觉
  • QT6 Python UI文件转换PY文件的方法
  • [硬件电路-111]:滤波的分类:模拟滤波与数字滤波; 无源滤波与有源滤波;低通、带通、带阻、高通滤波;时域滤波与频域滤波;低价滤波与高阶滤波。
  • Dify 从入门到精通(第 4/100 篇):快速上手 Dify 云端:5 分钟创建第一个应用