`
m635674608
  • 浏览: 5052191 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

为什么volatile能保证可见性?

    博客分类:
  • java
 
阅读更多

我们都知道volatile能保证可见性,不能保证原子性,比如i++操作

也知道Happen-Before原则,那么是如何确保Happen-Before原则不被指令重排序影响呢?(如果对上述描述有困惑请移步[高并发Java 三] Java内存模型和线程安全

例如你让一个volatile的integer自增(i++),其实要分成3步:1)读取volatile变量值到local; 2)增加变量的值;3)把local的值写回,让其它的线程可见。这3步的jvm指令为:

mov   
0xc(%r10),%r8d
 ; Load
inc   
 %r8d           ; Increment
mov   
 %r8d,0xc(%r10)
 ; Store
lock
 addl $0x0,(%rsp)
 ; StoreLoad Barrier

StoreLoad Barrier就是内存屏障

内存屏障(memory barrier) 是一个CPU指令。基本上,它是这样一条指令: a) 确保一些特定操作执行的顺序; b) 影响一些数据的可见性(可能是某些指令执行后的结果)。编译器和CPU可以在保证输出结果一样的情况下对指令重排序,使性能得到优化。插入一个内存屏障, 相当于告诉CPU和编译器先于这个命令的必须先执行,后于这个命令的必须后执行。内存屏障另一个作用是强制更新一次不同CPU的缓存。例如,一个写屏障会 把这个屏障前写入的数据刷新到缓存,这样任何试图读取该数据的线程将得到最新值,而不用考虑到底是被哪个cpu核心或者哪颗CPU执行的。

内存屏障和volatile什么关系?上面的虚拟机指令里面有提到,如果你的字段是volatile,Java内存模型将在写操作后插入一个写屏障 指令,在读操作前插入一个读屏障指令。这意味着如果你对一个volatile字段进行写操作,你必须知道:1、一旦你完成写入,任何访问这个字段的线程将 会得到最新的值。2、在你写入前,会保证所有之前发生的事已经发生,并且任何更新过的数据值也是可见的,因为内存屏障会把之前的写入值都刷新到缓存。 

明白了内存屏障这个CPU指令,回到前面的JVM指令:从Load到store到内存屏障,一共4步,其中最后一步jvm让这个最新的变量的值在所有线程可见,也就是最后一步让所有的CPU内核都获得了最新的值,但中间的几步(从Load到Store)是不安全的,中间如果其他的CPU修改了值将会丢失。

所以volatile不能保证i++操作的原子性

分享到:
评论

相关推荐

    4.4 volatile关键字有什么作用?.rar

    因此,多线程编程中,volatile用于标记那些可被多个线程同时访问并修改的变量,以保证线程间的可见性。 在GD32F303的源码中,我们可能会看到类似这样的例子: ```c volatile uint32_t flag; // 声明为volatile的...

    Java volatile与AQS锁内存可见性

    从JUC中的AQS引入,讲解Java volatile与AQS锁内存可见性

    Volatile详解,深入学习Volatile

    - volatile不能替代锁:尽管volatile可以保证可见性,但无法保证原子性。例如,两个线程同时读写一个volatile变量,可能导致数据不一致。 - volatile与指针:如果指针本身是volatile,意味着指针的值(地址)可能...

    Java并发volatile可见性的验证实现

    Java并发volatile可见性的验证实现 在Java并发编程中,volatile关键字扮演着非常重要的角色,它可以确保变量的可见性和原子性。在多线程环境下,volatile关键字可以确保变量的修改对其他线程是可见的。本文将通过...

    Java并行(3):可见性重访之锁、Volatile与原子变量1

    锁提供了严格的互斥和可见性,而`volatile`则是一种较弱的同步机制,只保证可见性。在设计并发程序时,开发者需要根据具体需求选择合适的方法来确保线程安全和数据一致性。理解这些概念有助于编写出高效且可靠的并发...

    java里的volatile关键字详解

    2. 原子性:volatile只能让被他修饰内容具有可见性,但不能保证它具有原子性。比如volatile int a=0;之后有一个操作a++;这个变量a具有可见性,但是a++依然是一个非原子操作,也就是这个操作同样存在线程安全问题。 3...

    volatile源码分析1

    本文将从JVM、C++以及汇编语言的角度深入探讨volatile的两大特性:禁止重排序和内存可见性。 1. 禁止重排序 重排序是指编译器和处理器为了优化性能而可能对指令序列进行重新排列的一种行为。在Java中,volatile...

    Java基础:volatile详解.pdf

    volatile保证可见性的作用是,多线程访问主内存的某一个资源时,如果某一个线程在自己的工作内存中修改了该资源,并写回主内存,那么JMM内存模型应该要通知其他线程来从新获取最新的资源,来保证最新资源的可见性。...

    深入解析volatile关键字:保障多线程下的数据一致性与可见性.pdf

    ### 深入解析volatile关键字:保障多线程下的数据一致性与可见性 #### 一、volatile的基本概念 ##### 1.1 可见性 在并发编程领域中,可见性是一个极为重要的概念,它指的是当一个线程修改了一个共享变量的值后,...

    18Java内存模型:Java中的volatile有什么用?1

    总之,`volatile`关键字在Java中起到了保证并发环境下变量可见性和有序性的作用,但并不能保证操作的原子性。在设计多线程程序时,理解`volatile`的工作原理及其限制对于正确地处理并发问题至关重要。

    Java并发:volatile内存可见性和指令重排

     1、保证内存可见性  2、防止指令重排  此外需注意volatile并不保证操作的原子性。  (一)内存可见性  1 概念  JVM内存模型:主内存和线程独立的工作内存  Java内存模型规定,对于多个线程共享的变量...

    C#中volatile与lock用法

    volatile是C#中用于控制同步的关键字,其意义是针对程序中一些敏感数据,不允许多线程同时访问,保证数据在任何访问时刻,最多有一个线程访问,以保证数据的完整性,volatile是修饰变量的修饰符。 1、volatile的使用...

    volatile 变量的说明

    总结来说,volatile是处理多线程并发和设备驱动编程时的重要工具,它确保了变量的实时性和可见性,但不能替代锁或其他同步机制来处理复杂的并发问题。理解和正确使用volatile是编写高效、可靠的并发程序的关键。

    volatile用法

    `volatile`关键字在C语言中是一个非常重要的概念,主要用于处理多线程环境或者与硬件交互时的内存可见性问题。它的主要作用是确保对变量的修改能够立即反映到所有线程,防止编译器进行不必要的优化,使得其他线程...

    Java并发编程:volatile关键字解析

    - **非原子性**:需要注意的是,虽然`volatile`关键字提供了可见性和有序性保证,但它并不能保证复合操作的原子性。例如,对于`i++`这样的操作,即使`i`是`volatile`变量,该操作也不是原子性的。 #### 五、使用...

    Java线程:volatile关键字

    volatile 变量具有 synchronized 的可见性特性,但是不具备原子特性。这就是说线程能够自动发现 volatile 变量的最新值。volatile 变量可用于提供线程安全,但是只能应用于非常有限的一组用例:多个变量之间或者某个...

    聊聊并发(一)深入分析Volatile的实现原理

    在并发编程领域,Volatile是Java中一个非常关键的特性,它为共享变量提供了内存可见性和有序性保证,但不保证原子性。本篇文章将深入分析Volatile的实现原理,结合`LinkedTransferQueue`和`TransferQueue`这两个与...

Global site tag (gtag.js) - Google Analytics