面试官:synchronized 能不能禁止指令重排序?大部分人都会答错!

这篇具有很好参考价值的文章主要介绍了面试官:synchronized 能不能禁止指令重排序?大部分人都会答错!。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

指令重排序

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)、字节码如下

面试官:synchronized 能不能禁止指令重排序?大部分人都会答错!

从字节码中可以看到,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开发手册(嵩山版)》最新发布,速速下载!

觉得不错,别忘了随手点赞+转发哦!文章来源地址https://www.toymoban.com/news/detail-679627.html

到了这里,关于面试官:synchronized 能不能禁止指令重排序?大部分人都会答错!的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 结对编程 --- 大部分程序员喜欢的编程方式

    一、介绍 结对编程起源时间可以追溯到 1990 年代早期。这种编程方法最初由 Jim Highsmith 和 Alistair Cockburn 等人提出。后来,Kent Beck 和 Ward Cunningham 等人将其发展成为一种敏捷开发方法,被称为“极限编程”(Extreme Programming,简称 XP)。结对编程是 XP 中的一种核心实践,也是

    2024年02月06日
    浏览(52)
  • 用Matlab实现车牌分割(可识别大部分蓝色、绿色车牌)

          最近学习了数字图像处理的腐蚀、膨胀、闭运算、开运算等内容,于是想进行实践。车牌分割是一个不错的选择,里面涉及到了很多知识点。       这里先简述一下车牌分割的思路和流程(这里以绿色车牌为例): 1.定位绿色车牌区域 2.车牌矫正(如果图像中车牌是倾

    2024年02月12日
    浏览(44)
  • 1200 + AI工具大收录,58个分类,支持大部分行业

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 一、使用步骤 总结 随着人工智能技术的不断发展,越来越多的AI工具涌现出来,它们在各个领域中得到了广泛的应用。除了常用的文本、图片、视频AI工具,还有普通办公、设计、编程、

    2024年02月16日
    浏览(37)
  • 低代码产品如何分类,大部分人都没有搞清楚

    最近许多技术峰会都出现了低代码这个名词,可以说,低代码是中台之后,又一个热门话题和名词了。 低代码平台是 无需编码或通过少量代码 就可以快速生成应用程序的开发平台。也是一款图形化、拖拉拽方式快速实现企业数字化转型中的创新应用、支持用少量代码扩展实

    2023年04月20日
    浏览(59)
  • MySQL 字段为 NULL 的5大坑,大部分人踩过

    在验证问题之前,我们先建一张测试表及测试数据。   构建的测试数据,如下图所示:   有了上面的表及数据之后,我们就来看当列中存在 NULL 值时,究竟会导致哪些问题? 我们都知道, count 是用来计数的,当表中某个字段存在 NULL 值时,就会造成 count 计算出来的数据丢

    2024年02月05日
    浏览(50)
  • 安全清理大部分的C盘内存(一般10GB以上)

     如果感觉有用请 关注,点赞,收藏!  下次分享更有用的干货~ 欢迎转载,请注明出处! 用360清理发现, windows search日志 占用了70多个G空间,先清除!    该日志文件有撒用呢?  如果没有这个日志文件,我们在文件系统进行搜索的时候就会比较慢了,而且还会出现这样的

    2023年04月15日
    浏览(54)
  • CTF Misc(2)内存取证基础以及原理,覆盖了大部分题型

    内存取证在ctf比赛中也是常见的题目,内存取证是指在计算机系统的内存中进行取证分析,以获取有关计算机系统当前状态的信息。内存取证通常用于分析计算机系统上运行的进程、网络连接、文件、注册表等信息,并可以用于检测和分析恶意软件、网络攻击和其他安全事件

    2024年02月12日
    浏览(47)
  • windows11(win10大部分通用)系统C盘清理 | 深度优化

    前言 :首先对于计算机来说,Windows操作系统一般是安装在磁盘驱动器的C盘中,运行时会产生许多  垃圾文件  ,C盘空间在一定程度上会越来越小。而把它作为生产力工具的我们,时间越久,C盘常常会提示显示其内存已不足。C盘容量不足将会极大  影响系统的运行速度  ,

    2024年02月05日
    浏览(75)
  • 校园综合服务平台V3.9.2 源码修复大部分已知BUG

    校园综合服务平台,版本更新至V3.9.1  ,源码功能强大,ui 精美, 功能包含但不限于校园跑腿,外卖,组局,圈子,商城,抽奖,投票,团购,二手市场,签到,积分商城,一元购等!即刻源码持续更新

    2024年04月26日
    浏览(55)
  • 50个Linux常用命令行快捷键(大部分适配Mac OS)

    50个Linux常用命令行快捷键 (大部分适配Mac OS) 移动光标到行首: Ctrl + a 移动光标到行尾: Ctrl + e 移动光标到上一个单词的开头: Ctrl + ← 移动光标到下一个单词的开头: Ctrl + → 删除光标之前的字符: Ctrl + u 删除光标之后的字符: Ctrl + k 删除光标之前的单词: Ctrl + w 清

    2024年02月10日
    浏览(61)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包