volatile关键字确保多线程环境下变量的可见性,1.当变量被修改时立即写回主内存,2.读取时强制从主内存获取最新值,3.禁止指令重排序,适用于标志位等场景,但不保证复合操作的原子性。

在Java中,volatile关键字用于确保变量的可见性,特别是在多线程环境下。当一个变量被声明为volatile,jvm会保证该变量在多个线程之间的修改是立即可见的,避免线程从各自的工作内存(缓存)中读取过期值。
volatile如何保证可见性
Java内存模型中,每个线程都有自己的工作内存,用于存储主内存中变量的副本。普通变量的读写操作可能只发生在工作内存中,导致一个线程修改了变量,其他线程无法及时看到变化。
而被volatile修饰的变量,有以下特性:
- 写操作:当一个线程修改了volatile变量,修改会立即刷新到主内存
- 读操作:当一个线程读取volatile变量时,会直接从主内存读取最新值,而不是使用本地缓存
- 禁止指令重排序:volatile变量的操作前后不会被JVM或CPU重排序,保障一定的有序性
使用场景示例
volatile适用于那些不需要原子性但需要及时可见的变量。常见用法是作为“标志位”控制线程运行状态。
立即学习“Java免费学习笔记(深入)”;
public class VolatileExample { private volatile boolean running = true; public void run() { while (running) { // 执行任务 } System.out.println("线程结束"); } public void stop() { running = false; // 其他线程调用stop后,run()中的循环能立即感知 } }
在这个例子中,如果没有volatile,主线程修改了running为false,工作线程可能一直读取的是自己缓存中的true值,导致无法退出循环。加上volatile后,修改对所有线程立即可见。
注意事项与局限性
volatile虽然能保证可见性,但不能保证复合操作的原子性。例如自增操作(i++)包含读、改、写三步,即使i是volatile,也不能保证整个过程线程安全。
- 适合用volatile的情况:变量的写入不依赖当前值,或者只有一个线程写,多个线程读
- 不适合的情况:涉及竞态条件的复合操作,应使用synchronized或Atomic类替代
基本上就这些。volatile是轻量级的同步机制,正确使用可以在不影响性能的前提下解决可见性问题。不复杂但容易忽略细节。


