volatile通过内存屏障保证变量可见性与有序性,每次读取从主内存加载、写入立即刷新主内存,并禁止指令重排,适用于状态标志控制、双重检查锁定等场景,但不保证复合操作的原子性,需配合synchronized或Atomic类使用。

在Java中,volatile关键字主要用于保证变量的可见性,是多线程编程中一种轻量级的同步机制。它不能替代synchronized,但在特定场景下非常高效。理解volatile的原理和使用技巧,有助于写出更安全、高效的并发程序。
volatile如何实现可见性
当一个变量被声明为volatile,JVM会确保该变量在多个线程之间的可见性。具体来说:
- 每次读取volatile变量时,都会从主内存中重新加载,而不是使用线程本地的缓存值。
- 每次写入volatile变量后,新值会立即刷新到主内存中。
- volatile变量的读写操作不会被重排序(禁止指令重排),这依赖于内存屏障(Memory Barrier)机制。
这种机制保证了一个线程对volatile变量的修改,其他线程能立刻看到,从而避免了因CPU缓存不一致导致的数据脏读问题。
适用场景:状态标志位控制
最常见的volatile使用场景是作为线程控制开关。例如,用一个boolean变量来控制某个线程是否继续运行:
立即学习“Java免费学习笔记(深入)”;
public class Worker {private volatile boolean running = true;
public void stop() {
running = false;
}
public void run() {
while (running) {
// 执行任务
}
System.out.println("Worker stopped");
}
}
这里如果不加volatile,主线程调用stop()修改running为false,工作线程可能因为读取的是缓存中的旧值而无法及时退出。
注意:volatile不保证原子性
volatile虽然保证了可见性和有序性,但不保证复合操作的原子性。例如下面的代码存在线程安全问题:
public class Counter {public volatile int count = 0;
public void increment() {
count++; // 非原子操作:读-改-写
}
}
count++包含三个步骤:读取count值、加1、写回。即使count是volatile的,多个线程同时执行increment仍可能导致丢失更新。此时应使用AtomicInteger或synchronized来保证原子性。
双重检查锁定中的volatile应用
在单例模式的双重检查锁定(Double-Checked Locking)实现中,volatile用于防止对象初始化过程中的指令重排序问题:
public class Singleton {private static volatile Singleton instance;
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
如果没有volatile,JVM在对象创建过程中可能发生指令重排,导致其他线程获取到一个未完全初始化的对象引用。加上volatile后,通过内存屏障阻止了这种重排序,确保安全性。
基本上就这些。volatile适用于状态标志、一次性安全发布等场景,关键是要清楚它的能力边界:保障可见性与有序性,但不解决原子性问题。合理使用,能有效提升性能又不失线程安全。










