上面介绍了使用锁的方式可以解决共享变量内存可见性问题,但是使用锁太笨重,因为它会带来线程上下文的切换开销。对于解决内存可见性问题,Java 还提供了一种弱形式的同步,也就是使用 volatile 关键字。该关键字可以确保对一个变量的更新对其他线程马上可见。当一个变量被声明为 volatile 时,线程在写入变量时不会把值缓存在寄存器或者其他地方,而是会把值刷新回主内存。当其他线程读取该共享变量时,会从主内存重新获取最新值,而不是使用当前线程的工作内存中的值。volatile 的内存语义和 synchronized 有相似之处,具体来说就是,当线程写入了 volatile 变量值时就等价于线程退出 synchronized 同步块(把写入工作内存的变量值同步到主内存),读取 volatile 变量值时就相当于进入同步块(先清空本地内存变量值,再从主内存获取最新值)。

    下面看一个使用 volatile 关键字解决内存可见性问题的例子。如下代码中的共享变量 value 是线程不安全的,因为这里没有使用适当的同步措施。

    1. public class ThreadNotSafeInteger {
    2. private int value;
    3. public int get() {
    4. return value;
    5. }
    6. public void set(int value) {
    7. this.value = value;
    8. }
    9. }

    首先来看使用 synchronized 关键字进行同步的方式。

    1. public class ThreadSafeInteger {
    2. private int value;
    3. public synchronized int get() {
    4. return value;
    5. }
    6. public synchronized void set(int value) {
    7. this.value = value;
    8. }
    9. }

    然后是使用 volatile 进行同步。

    1. public class ThreadSafeInteger {
    2. private volatile int value;
    3. public int get() {
    4. return value;
    5. }
    6. public void set(int value) {
    7. this.value = value;
    8. }
    9. }

    在这里使用 synchronized 和使用 volatile 是等价的,都解决了共享变量 value 的内存可见性问题,但是前者是独占锁,同时只能有一个线程调用 get()方法,其他调用线程会被阻塞,同时会存在线程上下文切换和线程重新调度的开销,这也是使用锁方式不好的地方。而后者是非阻塞算法,不会造成线程上下文切换的开销。

    但并非在所有情况下使用它们都是等价的,volatile 虽然提供了可见性保证,但并不保证操作的原子性。

    那么一般在什么时候才使用 volatile 关键字呢?

    ● 写入变量值不依赖变量的当前值时。因为如果依赖当前值,将是获取—计算—写入三步操作,这三步操作不是原子性的,而 volatile 不保证原子性。

    ● 读写变量值时没有加锁。因为加锁本身已经保证了内存可见性,这时候不需要把变量声明为 volatile 的。