三、volatile

2018-08-21  本文已影响0人  小绵羊你毛不多

特征

被volatile修饰的变量,具有两个特征

  1. 保证可见性
  2. 不保证原子性
  3. 禁止指令重排序

关于内存可见性、原子性、有序性,先来了解一下内存模型吧~

java内存模型(JMM)

特性
i = 0;  //是原子操作         
j = i ; //不是! 包含两个操作 1.读取i 2.赋值给j
i++;    //不是!三个操作 1.读取i 2.+1 3.赋值给i 

volatile是无法保证复合操作的原子性。想在多线程环境下保证原子性,可以通过锁、synchronized来确保

volatile可以保证可见性。当一个变量被volatile修饰之后,该变量被修改后立即更新到内存中,读取的时候会直接从内存中读取。

volatile禁止指令重排序,来保证一定的有序性

指令重排序:是JVM为了优化指令,提高程序运行效率,在不影响单线程程序执行结果的前提下,尽可能地提高并行度。注意是单线程,多线程情况下会有问题啊

原理

在jvm底层 是采用‘内存屏障’来实现的

应用场景

int a = 0;
//修改后立刻对线程可见 比sync lock有一定的效率提升
volatile bool flag = false;

public void write() {
    a = 2;              //1
    flag = true;        //2
}

public void multiply() {
    if (flag) {         //3
        int ret = a * a;//4
    }
}
懒汉模式
class Singleton{
//为了避免初始化操作的指令重排序 
    private volatile static Singleton instance = null;
 
    private Singleton() {
 
    }
 
    public static Singleton getInstance() {
        if(instance==null) { //B
            synchronized (Singleton.class) {
                if(instance==null)
//在Singleton构造函数体执行之前,变量instance可能成为非null!
                    instance = new Singleton(); //A
            }
        }
        return instance;
    }
}
1.线程1进入到//A处,但在构造函数执行之前。使实例成为非null
2.线程2进入//B处,实例不为null,将instance引用返回。返回了一个构造完整但部分初始化的singleton对象
 public volatile String lastUser; //发布的信息
 
    public boolean authenticate(String user, String password) {
        boolean valid = passwordIsValid(user, password);
        if (valid) {
            User u = new User();
            activeUsers.add(u);
            lastUser = user;
        }
        return valid;
    }

private volatile int value;
 
    //读操作,没有synchronized,提高性能
    public int getValue() { 
        return value; 
    } 
 
    //写操作,必须synchronized。因为x++不是原子操作
    public synchronized int increment() {
        return value++;
    }

上一篇下一篇

猜你喜欢

热点阅读