指令重排序

3 篇文章 0 订阅

1. 代码

/**
 * @author cxyxh
 * @date 2021-02-26
 */
public class TestInstructionsSwap {

    private static int a = 0, b = 0;
    private static int x = 0, y = 0;

    public static void main(String[] args) throws InterruptedException {
        int i = 0;
        for (;;){
            a = 0;
            b = 0;
            x = 0;
            y = 0;

            Thread thread1 = new Thread(() -> {
                a = 1;
                x = b;
            });

            Thread thread2 = new Thread(() -> {
                b = 1;
                y = a;
            });

            thread1.start();
            thread2.start();
            thread1.join();
            thread2.join();
            
            if (y == 0 && x == 0){
                System.out.println("确定是有指令重排序,循环次数:" + i);
                break;
            }
            i++;
        }
    }
}

执行结果

image-20210228194333220

每次执行的结果不一样,有可能会等很久。

2. 解释

首先假设如果不会发生指令重排序,那么这个程序的执行结果可能有哪些呢?

假设没有指令重排序的话,因为至少会有一个线程先执行,也就是说,至少x,y中至少有一个应该是1,而不会出现两个都为0的情况。那么我们就可以得出,如果出现了x = 0 并且 y = 0的情况,那么就说明出现了指令重排序。

详细情况:

(1)thread1先执行

image-20210228194902646

最终结果,第一个结果是(x = 0,y = 1),第二个结果是(x = 1,y = 1),第三个结果是(x = 1,y = 1)。

(2)thread2先执行

image-20210228195324857

第一个结果是(x = 1,y = 0),第二个结果是(x = 1,y = 1),第三个结果是(x = 1,y = 1)。

那么我们执行代码的最终结果是出现了x和y都等于0的情况,那么说明代码执行过程中还是出现了指令重排序。

3. 其他

为什么会出现指令重排序?目的是为了提高性能。

重排序分为3种:

(1)编译器优化的重排序:

在编译期间,编译器会在不改变单线程程序语义的前提下,重新安排语句的执行顺序。

(2)指令级并行的重排序:

现代处理器采用了指令级并行技术来讲多条指令重叠执行。如果不存在数据依赖性,处理器可以改变语句对应机器指令的执行顺序。

(3)内存系统的重排序:

由于处理器使用了缓存和读/写缓冲区,使得加载和存储操作看上去可能是在乱序执行,因为从缓冲区到写入内存,需要的时间不一致。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值