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

作者:绿仔牛奶_ 时间:2023-07-12 03:07:58 

什么是有序性

在开发中,我们通常按照从上到下的顺序编写程序指令,并且希望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,但是它可能还没有初始化但是却指向了某片内存空间。

我们就下图进行分析

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

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

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

解决有序性?

  • Volatile修饰保证有序性

  • 使用Synchtonized加锁保证有序性

  • 使用Lock加锁保证有序性

来源:https://blog.csdn.net/yuqu1028/article/details/128680948

标签:Java,线程安全,有序性
0
投稿

猜你喜欢

  • SpringBoot集成ElasticSearch的示例代码

    2023-05-02 05:50:28
  • Spring实现处理跨域请求代码详解

    2023-11-25 12:28:34
  • C#命令模式用法实例

    2021-10-21 12:46:02
  • Java实现读取及生成Excel文件的方法

    2023-10-06 04:21:27
  • 在web.config和app.config文件中增加自定义配置节点的方法

    2021-11-03 03:21:41
  • Android中通过view方式获取当前Activity的屏幕截图实现方法

    2021-09-20 09:17:22
  • C#复制数组的两种方式及效率比较

    2023-07-15 04:19:12
  • 一篇文章让你彻底了解Java可重入锁和不可重入锁

    2023-12-06 11:57:26
  • SpringBoot的异常处理流程是什么样的?

    2021-07-09 17:54:40
  • 详解Android中Service AIDL的使用

    2022-10-02 05:08:16
  • C语言实现生日贺卡

    2023-03-04 18:12:05
  • Android仿京东分类模块左侧分类条目效果

    2022-09-09 04:35:34
  • c# 用ELMAH日志组件处理异常

    2022-03-11 12:05:36
  • Hadoop+HBase+ZooKeeper分布式集群环境搭建步骤

    2022-02-13 01:16:33
  • SpringBoot结合Vue实现投票系统过程详解

    2022-08-24 16:12:30
  • c#中XML解析文件出错解决方法

    2022-01-21 00:38:50
  • Spring @Transactional注解失效解决方案

    2022-10-25 05:30:30
  • java中String.intern()方法功能介绍

    2023-11-25 23:49:21
  • C#中跨线程访问控件问题解决方案分享

    2021-06-27 18:47:24
  • spring BeanProcessor接口详解

    2021-12-26 07:32:17
  • asp之家 软件编程 m.aspxhome.com