Java线程安全中的有序性浅析

 更新时间:2023年02月21日 10:23:29   作者:绿仔牛奶_  
这篇文章主要介绍了Java线程安全中的有序性,在开发中,我们通常按照从上到下的顺序编写程序指令,并且希望cpu和编译器按照我们预先编写的顺序去执。但往往cpu和编译器为了提高性能、优化指令的执行顺序,会将我们编写好的程序指令进行重排序

什么是有序性

在开发中,我们通常按照从上到下的顺序编写程序指令,并且希望cpu和编译器按照我们预先编写的顺序去执。但往往cpu和编译器为了提高性能、优化指令的执行顺序,会将我们编写好的程序指令进行重排序。

此时如果是在单线程状态下,无论是否进行了重排序都不会影响程序最终的结果

而有序性是指在多线程环境下就可能会由于程序指令重排序后导致最终结果与预期不符的情况

我们以单例模式中的双重检验锁为例

利用new关键字创建一个对象实际上是执行了三个操作

  • 分配内存空间
  • 在内存上(执行构造方法)初始化对象
  • 将初始化后的对象提交给引用(对象引用指向分配好的内存空间地址)

但是当我们在运行程序时,编译器对程序进行重排序优化,经常会将2和3两个步骤进行调换。

// 双重检验锁
public class Singleton {
    static Singleton instance;
    static Singleton getInstance(){
        if (instance == null){
            synchronized(Singleton.class){
                if (instance == null){
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

上述双重检验锁,在第一次校验instance是否为null时如果不为null,则不用进行后续的初始化的下面的加锁操作,大幅的提高了synchronized的性能。但是在多线程状态下执行上述创建对象的程序,就可能会出现创建的对象instance虽然不为null,但是它可能还没有初始化但是却指向了某片内存空间。

我们就下图进行分析

我们假设A和B两条线程同时创建对象,那么上述的A线程创建instance时为其分配内存空间,正确来讲应该先对instance进行初始化然后将内存地址交给instance,但是由于重排序,却在初始化之前提交了内存地址。那么当线程切换到B,B就会认为instance是一个创建完成的对象就会返回。

双重检验锁的有序性就体现在,创建对象的三个操作被重排序之后可能执行顺序会变成先提交内存地址再初始化导致对象创建失败

解决有序性?

  • Volatile修饰保证有序性
  • 使用Synchtonized加锁保证有序性
  • 使用Lock加锁保证有序性

到此这篇关于Java线程安全中的有序性浅析的文章就介绍到这了,更多相关Java线程有序性内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java Socket编程心跳包创建实例解析

    Java Socket编程心跳包创建实例解析

    这篇文章主要介绍了Java Socket编程心跳包创建实例解析,具有一定借鉴价值,需要的朋友可以参考下
    2017-12-12
  • 使用JAXBContext 设置xml节点属性

    使用JAXBContext 设置xml节点属性

    这篇文章主要介绍了使用JAXBContext 设置xml节点属性的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • java对double数组排序示例分享

    java对double数组排序示例分享

    这篇文章主要介绍了java对double数组排序示例,代码简单,下面我们直接上代码,需要的朋友可以参考下
    2014-03-03
  • Java StringUtils字符串分割转数组的实现

    Java StringUtils字符串分割转数组的实现

    这篇文章主要介绍了Java StringUtils字符串分割转数组的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • Java23种设计模式中的单例模式你了解吗

    Java23种设计模式中的单例模式你了解吗

    这篇文章主要为大家详细介绍了Java23种设计模式中的单例模式,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-02-02
  • 使用SpringAOP获取用户操作日志入库

    使用SpringAOP获取用户操作日志入库

    这篇文章主要介绍了使用SpringAOP获取用户操作日志入库,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • 通过简单方法实现spring boot web项目

    通过简单方法实现spring boot web项目

    这篇文章主要介绍了通过简单方法实现spring boot web项目,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-09-09
  • Java使用HashMap映射实现消费抽奖功能

    Java使用HashMap映射实现消费抽奖功能

    这篇文章主要为大家详细介绍了Java使用HashMap映射实现消费抽奖功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • MyEclipse+Tomcat+MAVEN+SVN项目完整环境搭建(图文教程)

    MyEclipse+Tomcat+MAVEN+SVN项目完整环境搭建(图文教程)

    这篇文章主要介绍了MyEclipse+Tomcat+MAVEN+SVN项目完整环境搭建(图文教程),非常具有实用价值,需要的朋友可以参考下
    2017-12-12
  • Spring Boot+微信小程序开发平台保存微信登录者的个人信息

    Spring Boot+微信小程序开发平台保存微信登录者的个人信息

    这篇文章主要介绍了Spring Boot+微信小程序开发平台保存微信登录者的个人信息,本文主要介绍 wx.login和wx.getProfile接口,因篇幅所限,不能对其它接口做详细介绍 ,有兴趣者可以查阅官方文档
    2022-05-05

最新评论