指令重排序
在多线程并发编程的过程中,执行重排序有时候会造成错误的后果,比如一个线程在main线程中调用setFlag(true)的前边修改了某些程序配置项,而在t1线程里需要用到这些配置项,所以会造成配置缺失的错误。但是java给我们提供了一些抑制指令重排序的方式。
1.同步代码抑制指令重排序
将需要抑制指令重排序的代码放入同步代码块中:

在获取锁的时候,它前边的操作必须已经执行完成,不能和同步代码块重排序;在释放锁的时候,同步代码块中的代码必须全部执行完成,不能和同步代码块后边的代码重排序。
2.volatile变量抑制指令重排序
public class Reordering {
private static volatile boolean flag;
private static int num;
public static void main(String[] args) {
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
while (!flag) {
Thread.yield();
}
System.out.println(num);
}
});
t1.start();
num = 5;
flag = true;
}
}
具体抑制重排序的规则如下:
-
volatile写之前的操作不会被重排序到volatile写之后。
-
volatile读之后的操作不会被重排序到volatile读之前。
-
前边是volatile写,后边是volatile读,这两个操作不能重排序。
3.final变量抑制指令重排序
具体的规则就这两条:
-
在构造方法内对一个final字段的写入,与随后把这个被构造对象的引用赋值给一个引用变量,这两个操作之间不能重排序。
-
初次读一个包含final字段对象的引用,与随后初次读这个final字段,这两个操作不能重排序。
浙公网安备 33010602011771号