8.volatile为啥不能保证原子性?

简介: 8.volatile为啥不能保证原子性?

volatile为什么不能保证原子性


小陈:老王,快来快来,上一篇结尾说volatile不能保证原子性,我现在迫不及待了...

老王:哈哈,来了,马上开搞......

老王:按照惯例,我还是先来给你画张图:

还是以 i++ 的那个例子为例,volatile int i = 0,假如两个线程A线程B同时对 i 进行 ++ 操作如下:

image.png

上图存在一种情况就是,线程A线程B如果几乎同时读取 i = 0 到自己的工作内存中。

线程A执行 i++  结果后将 i = 1 赋值给工作内存;但是这个时候还没来的将最新的结果刷新回主内存的时候线程B读取主内存的旧值 i = 0 ,然后执行use指令将 i = 0的值传递给线程B去进行操作了。

即使这个时候线程A立即将 i = 1刷入主内存那也晚了线程B已经使用旧值 i = 0进行操作了,像这种情况计算结果就不对了。

老王:小陈,我上面的那个图讲解,你可以听懂嘛?

小陈:嗯嗯,看图解释就是,线程Ai ++  结果, 也就是 i = 1还没刷回主内存线程B就执行 use指令将 i = 0传递给cpu了 ,导致线程B使用的就旧的值 i = 0去进行操作,得到结果是错的。


怎样才能保证原子性?


小陈:那如果要保证原子性,应该是怎么样子的?

老王:如果要保证原子性的话,落到底层实际还是需要进行加锁的,需要保证任意时刻只能有一个线程能执行成功。

比如在硬件层次或者对总线进行加锁,使得某一时刻只能有一个线程能执行i++ 操作,这样才能是不被中断的,才是原子性的。

现在现在这种情况,相当于就是两个线程同时进行了 i++操作线程Ai++ 操作还没结束;线程Bi++ 操作就也同时进行着,这种情况不是原子的。

小陈:哦,是不是可以这么理解:

如果要保证原子性的话,同一时刻只能有一个线程或者CPU能够执行成功底层是需要对硬件进行加锁的,只有某个CPU或者线程锁定了,享有独占的权限,那么它的操作才能是不被其它CPU或者线程打断的。

老王:没错,就是这个道理;你只有在硬件级别加锁了之后,享有独占的权限你的操作才能是不被其它CPU或线程打断的。

小陈:好的,老王,这么说我就理解了。

老王:这一篇对volatile不能保障原子性的解释,你再多看看几遍,多理解一下。再对之前的文章再复习复习,包括内存屏障、java内存模型、MESI一致性协议等知识,从下一章看是我们就要进入新的学习了。

小陈:好啊老王,下面的篇章我们要学习什么知识啊?

老王:下面我们就要进入synchronized的学习了....

相关文章
|
6月前
|
安全 Java 编译器
【面试问题】说说原子性、可见性、有序性?
【1月更文挑战第27天】【面试问题】说说原子性、可见性、有序性?
|
6月前
|
缓存
15.unsafe类的CAS是怎么保证原子性的?
15.unsafe类的CAS是怎么保证原子性的?
65 0
15.unsafe类的CAS是怎么保证原子性的?
|
4月前
|
安全 Java
Volatile不保证原子性及解决方案
**原子性在并发编程中确保操作不可中断,保持数据一致性。volatile保证可见性但不保证原子性,如`count++`在多线程环境下仍可能导致数据不一致。解决方案包括使用`synchronized`、`AtomicInteger`或`ReentrantLock`来确保复合操作的原子性和线程安全。例子展示了volatile在并发自增中的局限性,实际值通常小于预期,强调了正确选择同步机制的重要性。**
|
6月前
|
缓存 Java 编译器
面试官:volatile如何保证可见性的,具体如何实现?
面试官:volatile如何保证可见性的,具体如何实现?
73 0
|
6月前
|
缓存 安全 Java
多线程的三大特性:原子性、可见性和有序性
多线程的三大特性:原子性、可见性和有序性
133 0
|
6月前
|
缓存 安全 Java
5.volatile是什么?怎么保证可见性?
5.volatile是什么?怎么保证可见性?
72 0
5.volatile是什么?怎么保证可见性?
|
6月前
|
缓存 Java
13.synchronized总结:怎么保证可见性、有序性、原子性?
13.synchronized总结:怎么保证可见性、有序性、原子性?
99 0
13.synchronized总结:怎么保证可见性、有序性、原子性?
|
6月前
|
安全 Java
7.volatile怎么通过内存屏障保证可见性和有序性?
7.volatile怎么通过内存屏障保证可见性和有序性?
58 0
7.volatile怎么通过内存屏障保证可见性和有序性?
|
6月前
|
Java
18.AtomicReference、AtomicStampReference底层原理。多个变量更新怎么保证原子性?CAS的ABA问题怎么解决?
18.AtomicReference、AtomicStampReference底层原理。多个变量更新怎么保证原子性?CAS的ABA问题怎么解决?
51 0
18.AtomicReference、AtomicStampReference底层原理。多个变量更新怎么保证原子性?CAS的ABA问题怎么解决?
|
6月前
|
存储 安全 Java
关于volatile解决内存可见性问题(保证线程安全)
关于volatile解决内存可见性问题(保证线程安全)