关于volatile的理解

经常会听到volatile这个关键字,但没有深入的去了解过它,今天好好的整理一下
要谈volatile,我们先谈谈它的老大哥synchronized
一.synchronized

并发编程中最重要的三个特性是什么?原子性,可见性,有序性。只要有一个不能保证,就有可能导致程序的运行错误,我们熟知的synchronized就能保障原子性,可见性,有序性,因为synchronized能保障任意一个时刻只有一个线程执行该代码块,自然就不存在原子性的问题,又因为在释放锁之前会将变量的修改刷新到主存中,因此保证可见性,又因为每一时刻只有一个线程在执行代码,相当于让线程顺序执行同步代码,所以也可以保证有序性。所以synchronized可保证原子性,可见性,有序性

二.volatile

volatile可以说是一个稍弱的同步机制,因为它可以保障可见性和有序性,不能保障原子性

1.volatile可见性

先看下图,java中所有的变量都存储在主内存区,在多线程环境中,还有自己的工作内存,线程操作变量时是从主存中拷贝变量到工作内存操作,这样很容易产生"脏读"的问题。synchronized解决此问题的方法上面也谈到了,volatile是如何保证的呢?当一个变量被volatile修饰时,它会保证每次被修改的值会被立刻更新到主存,而且当缓存1对某个变量进行修改时,其它线程的工作内存中的该变量的缓存行无效(反映到硬件层的话,就是CPU的L1或者L2缓存中对应的缓存行无效),缓存2发现自己的变量无效后会等待缓存行对应的主存地址被更新后从主存中读取最新值,这样就保证了可见性

关于volatile的理解_第1张图片

2.volatile有序性

说到volatile的有序性就要提到指令重排序(Instruction Reorder),什么是指令重排序?这是JVM为了提高代码的执行效率对代码进行的优化,它不能保证代码先后顺序执行的一致,但可以保证执行的结果和顺序执行的结构一致。但在多线程环境中就可能出现问题,如下面代码

        //线程1
        func1();               语句1
        boolean flag = true;   语句2

        //线程2
        while (!flag){
            sleep();
        }
        func2();

线程1中的语句1和语句2并没有数据依赖关系,所以可能会进行指令重排序,先去执行语句2,而这时线程2会以为线程1已经执行完func1()而去执行func2(),这样就导致程序出错。
volatile关键字修饰的变量能禁止指令重排序,这样就在一定程度上保证了有序性
应用:

(1)状态标记量
        //线程1
        func1();                        语句1
        boolean volatile flag = true;   语句2

        //线程2
        while (!flag){
            sleep();
        }
        func2();
这是对上面程序的改进,避免了程序出错

(2)单例模式中的双重校验锁

public class Singleton3 {
    private Singleton3(){}
    private static volatile Singleton3 instance = null;

    public static Singleton3 getInstance() {
        if(instance == null){
            synchronized (Singleton3.class){
                instance = new Singleton3();
            }
        }
        return instance;
    }
}

我们看到已经有synchronizd来保证线程同步,为什么还需要volatile来修饰变量instance呢?
原因在 instance = new Singleton3();这句,JVM实际上对它进行了如下操作

(1)给instance分配内存
(2)调用Singleton3的构造函数完成初始化成员变量
(3)将instance指向内存分配空间

但由于JVM存在指令重排序优化,执行的顺序可能就为1-3-2,就可能发生当3执行完另一个线程以为它已经执行完了,抢占当前线程,这时instance非null但没有初始化,所以另一个线程和以后的线程都会直接返回该没有初始化的instance,从而出现错误。当有volatile修饰变量instance,就可以禁止指令重排序,从而避免出错。

你可能感兴趣的:(java,volatile)