关于java:面试官synchronized-能不能禁止指令重排序大部分人都会答错

1次阅读

共计 2561 个字符,预计需要花费 7 分钟才能阅读完成。

指令重排序

1、问题形容

首先肯定要明确:指令重排序和有序性是不一样的。这一点十分重要。

咱们常常都会这么说:

  • volatile 能保障内存可见性、禁止指令重排序然而不能保障原子性。
  • synchronized 能保障原子性、可见性和有序性。

留神:这里的有序性并不是代表能禁止指令重排序。

举个例子:

在双重查看的单例模式中,既然曾经加了 synchronized 为什么还须要 volatile 去润饰变量呢?如果 synchronized 能禁止指令重排,那么齐全能够不必要 volatile。

举荐一个开源收费的 Spring Boot 实战我的项目:

https://github.com/javastacks/spring-boot-best-practice

2、DCL 代码字节码剖析指令重排序问题

首先须要晓得的知识点:Object obj = new Object(); 这句代码并不是一个原子操作,他分为三步:

  • 在内存申请一片空间,new 出一个对象
  • 调用 new 出的这个对象的构造方法
  • 将这个对象的援用赋值给 obj
a)、DCL 双重查看代码
public class MySingleton {

    private static MySingleton INSTANCE;

    private MySingleton() {}

    public static MySingleton getInstance() {if (INSTANCE == null) {synchronized (MySingleton.class) {if (INSTANCE == null) {INSTANCE = new MySingleton();
                }
            }
        }
        return INSTANCE;
    }
}
b)、字节码如下

从字节码中能够看到,new MySingleton();这句代码对应了 17、20、21、24 这四行字节码(20 行是一个援用的拷贝,能够疏忽)。

  • 首先在 17 行在内存中开拓一块空间创立一个 MySingleton 对象。
  • 而后在 21 行调用该对象的构造方法。
  • 而后在 24 即将该对象的援用赋值给动态变量 INSTANCE。

以上是咱们冀望的执行程序,咱们心愿每个线程都依照该程序去执行指令(这就是禁止指令重排序)。然而因为计算机为了进步运行效率,会将咱们的指令程序进行优化重排(比方下面的程序可能会优化重排为:17、24、21)

指令重排序带来的问题

  • 咱们的计算机为了晋升效率,会将咱们的代码程序做一些优化,比方在 t1 线程中的执行程序是 17、24、21,在 t2 线程中执行的程序是 17、21、24(在单个线程中不论是那种执行程序都不会有问题)。
  • 当 t1 线程获取到锁执行对象创立的时候,先执行了 24 行,将该对象的援用赋值给了动态变量 INSTANCE(此时对象还没调用构造方法,该对象还不是一个残缺的对象)。
  • 此时 t2 线程开始运行了,当 t2 线程执行到 if (INSTANCE == null)(第 16 行代码) 语句的时候,t2 线程发现 INSTANCE 不为空,此时 t2 线程间接返回 INSTANCE 对象。然而此时该对象还是一个不残缺的对象,在 t2 线程应用该对象的时候就会呈现问题。

所以说指令重排序在单线程中是不会有任何问题的,然而一旦波及到多线程的状况,那么指令重排序可能会带来意想不到的后果。

有序性

那么既然 synchronized 不能禁止指令重排序,那么他保障的有序性是什么有序呢?

它的实质是让多个线程在调用 synchronized 润饰的办法时,由并行(并发)变成串行调用,谁取得锁谁执行。

1、代码示例

t1、t2 两个线程都须要去获取单例对象,而后调用 test 办法,并且 test 办法是加了同步锁的办法。

public class MySingleton {

    private static MySingleton INSTANCE;

    private MySingleton() {}

    public static MySingleton getInstance() {if (INSTANCE == null) {synchronized (MySingleton.class) {if (INSTANCE == null) {INSTANCE = new MySingleton();
                }
            }
        }
        return INSTANCE;
    }

    public static void test(final MySingleton singleton) {synchronized (MySingleton.class) {System.out.println(singleton);
        }
    }
}

测试代码

public class MySingletonTest {
  // 能够看到两个线程都须要去获取单例对象,而后调用 test 办法,并且 test 办法是加了同步锁的办法
    public static void main(final String[] args) {new Thread(() -> {MySingleton instance = MySingleton.getInstance();
            MySingleton.test(instance);
        }, "t1").start();
        new Thread(() -> {MySingleton instance = MySingleton.getInstance();
            MySingleton.test(instance);
        }, "t2").start();}
}

即便是 t2 线程取得了未调用构造函数的对象,那么在 t2 线程中再去调用 MySingleton.test(instance); 办法的时候,也并不会呈现任何问题,因为应用了同步锁,每个一加锁执行的办法都变成了串行,将并发执行变成了串行,当 t2 线程获取到锁而后执行的时候,t1 早曾经开释了锁,此时 instance 也曾经早就被实例化好了。所以不会呈现问题。

所以 synchronized 保障程序性是指的将并发执行变成了串行,但并不能保障外部指令重排序问题。

起源:blog.csdn.net/Hellowenpan/article/details/117750543

近期热文举荐:

1.1,000+ 道 Java 面试题及答案整顿(2022 最新版)

2. 劲爆!Java 协程要来了。。。

3.Spring Boot 2.x 教程,太全了!

4. 别再写满屏的爆爆爆炸类了,试试装璜器模式,这才是优雅的形式!!

5.《Java 开发手册(嵩山版)》最新公布,速速下载!

感觉不错,别忘了顺手点赞 + 转发哦!

正文完
 0