volatile写操作的happens-before规则

当一个线程对volatile变量进行写操作时:

  • JMM会确保该写操作之前的所有操作(包括普通变量的读写)都已完成,并且这些操作的结果对后续的volatile读操作是可见的。

  • 这意味着,volatile写操作之前的操作不会被重排序到写操作之后。

示例:

java

复制

int x = 1;
volatile boolean flag = false;

// 线程1
x = 2;          // 普通写操作
flag = true;    // volatile 写操作

在以上代码中:

  • x = 2的写操作一定会在flag = true之前完成。

  • flag = true写入主内存后,其他线程可以看到x = 2的结果。


2. volatile读操作的happens-before规则

当一个线程对volatile变量进行读操作时:

  • JMM会确保该读操作之后的所有操作(包括普通变量的读写)都不会被重排序到读操作之前

  • 这意味着,volatile读操作之后的操作可以看到volatile变量的最新值。

示例:

java

复制

// 线程2
if (flag) {       // volatile 读操作
    System.out.println(x); // 普通读操作
}

在以上代码中:

  • flag的读操作会从主内存中加载最新值。

  • System.out.println(x)的读操作一定会在flag的读操作之后执行,并且可以看到x的最新值。


3. volatilehappens-before关系

volatile变量的读写操作建立了线程之间的happens-before关系:

  • 写操作happens-before读操作:如果一个线程写入了volatile变量,另一个线程读取了该变量,那么写操作之前的所有操作对读操作之后的代码是可见的。

示例:

java

复制

// 线程1
x = 42;          // 普通写操作
flag = true;     // volatile 写操作

// 线程2
if (flag) {      // volatile 读操作
    System.out.println(x); // 普通读操作
}

在以上代码中:

  • 线程1的x = 42flag = true之间存在happens-before关系。

  • 线程2的flag读操作和System.out.println(x)之间也存在happens-before关系。

  • 因此,线程2可以确保看到x = 42的结果。


4. volatile与指令重排序

volatile不仅保证了可见性,还通过禁止指令重排序来确保程序的有序性:

  • 写操作volatile写操作之前的操作不会被重排序到写操作之后。

  • 读操作volatile读操作之后的操作不会被重排序到读操作之前。


5. 总结

  • 写操作volatile写操作会确保之前的所有操作都已完成,并将结果刷新到主内存。

  • 读操作volatile读操作会从主内存加载最新值,并确保之后的操作不会被重排序到读操作之前。

  • happens-before关系volatile变量的读写操作建立了线程之间的happens-before关系,确保了多线程环境下的可见性和有序性。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值