Redian新闻
>
对线面试官:synchronized 到底能不能禁止指令重排序?

对线面试官:synchronized 到底能不能禁止指令重排序?

公众号新闻

点击关注公众号,Java干货及时送达👇

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

  • I)、指令重排序
    • 1、问题描述
    • 2、DCL代码字节码分析指令重排序问题
  • II)、有序性
    • 1、代码示例

I)、指令重排序

1、问题描述

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

我们经常都会这么说:

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

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

举个例子:

在双重检查的单例模式中,既然已经加了synchronized为什么还需要volatile去修饰变量呢?如果synchronized能禁止指令重排,那么完全可以不用要volatile。

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线程使用该对象的时候就会出现问题。

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

II)、有序性

那么既然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保证顺序性是指的将并发执行变成了串行,但并不能保证内部指令重排序问题。

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
开源 Java 性能分析器比较:VisualVM、JMC 和 async-profiler我与孙儿比童年 冷明维立克前面试官告诉你--如何取得“顶级美高预面试”维立克面试高分?国人为什么爱抱团夏天剧烈运动后到底能不能喝冰水?今晚直播 | 上海财经大学博士生陈畅欣:学习重排中的深层次排序表示Lenovo Legion 5 Pro 16in 165Hz QHD IPS NVIDIA G-SYNC R7-5800H 16上门按摩APP,到底能不能用?Asus ROG Strix XG276Q 27" FHD 170Hz 1ms G-Sync Gaming Monitor小喇叭,广播啦大胆发言:偶像到底能不能恋爱?面试官:业务开发时,接口不能对外暴露怎么办?我有 3 种实现方案!加国召回6种这类饮料 到底能不能喝?献血到底能不能降低心血管疾病风险?他们“吵”起来了……丨医起推理吧中式主题乐园,到底能不能赚?失眠到底能不能用“曲唑酮”?冷却的不止季节(80)— 手术过后澳女发现买来的鸡蛋上长满颗粒状凸起,专家:到底能不能吃?还没出生就被认定是「暴力狂」,超雄宝宝到底能不能要?nǚ hóng?nǚ gōng全网抵制的预制菜,到底能不能给孩子吃?包爸说点实在的!Logitech G G512 CARBON LIGHTSYNC RGB Mechanical Gaming Keyboard 窒息!毕业展对战“熊孩子”大失败!数十个作品被摔、被撕、被毁坏!网友:求求!能不能禁止未成年人进入啊?印花税到底能不能调降或者取消?对市场一定会是利好吗面试官:Redis 缓存满了怎么办?她到底能不能演古偶?150刀Logitech G560 LIGHTSYNC PC Gaming Speaker头皮发麻! Costco三文鱼, 惊现活虫蠕动! 到底能不能生吃? 员工回应…async/await 函数到底要不要加 try catch ?玫瑰谷歌新动作,ChromeOS将不再是原来的ChromeOS教育部明确回应预警期刊到底能不能投?Logitech G815 Lightsync RGB(one back foot missing)不止有氚,日本核污水到底能不能喝?为什么不能禁止孩子用手机?这是我见过超好的答案
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。