双重检查锁单例与内存屏障分析

简介: 双重检查锁单例与内存屏障分析

单例(双重检查锁)


public class DoubleCheckLockSingleton {
    private static DoubleCheckLockSingleton instance = null;
    private DoubleCheckLockSingleton(){
    }
    public static DoubleCheckLockSingleton getInstance() {
        if (instance == null){
                        /**
             * 10 monitorenter
             * 11 getstatic #2 <DoubleCheckLockSingleton.instance : LDoubleCheckLockSingleton;>//获取静态变量
             * 14 ifnonnull 27 (+13)//判空
             * 17 new #3 <DoubleCheckLockSingleton>//如果等于null,创建对象
             * 20 dup
             * 21 invokespecial #4 <DoubleCheckLockSingleton.<init> : ()V>//对象的初始化
             * 24 putstatic #2 <DoubleCheckLockSingleton.instance : LDoubleCheckLockSingleton;>//给变量赋值
             * 27 aload_0
             * 28 monitorexit
             */
            synchronized (DoubleCheckLockSingleton.class){
                if (instance == null){
                    instance = new DoubleCheckLockSingleton();
                }
            }
        }
        return instance;
    }
    public static void main(String[] args) {
        DoubleCheckLockSingleton doubleCheckLockSingleton = DoubleCheckLockSingleton.getInstance();
    }
}


程序员称其为双重检查锁,因为会有两次检查 instance == null,一次是在同步块外,一次是在同步块内。为什么在同步块内还要再检验一次?因为可能会有多个线程一起进入同步块外的 if,如果在同步块内不进行二次检验的话就会生成多个实例了。注意上面的代码中属性是没有加volatile关键字的,是有可能发生指令重排的,


对于JVM来说:instance=new DoubleCheckLockSingleton();是做了以下的三件事的:


  1. 给 instance 分配内存
  2. 调用 Singleton 的构造函数来初始化成员变量
  3. 将instance对象指向分配的内存空间(执行完这步 instance 就为非 null 了)

对面上面的这段代码来说


JVM执行到instance=new DoubleCheckLockSingleton()时,可能的实行顺序就是1–>2–>3或1–>3–2,再多线程的情况下上面的两种情况都是有可能出现的。如果是前者自然是没有什么问题,但是如果出现后者那么就没有按照我们希望的循序执行。这样会出现已经重复创建对象的情况,也就没有达到我们单例的设计原则。


从更底层的方式来思考双锁出现的问题


这里我们使用idea插件jclasslib来分析(字节码阅读器)


主要看下面这部分:


 /**
 * 10 monitorenter
 * 11 getstatic #2 <DoubleCheckLockSingleton.instance : LDoubleCheckLockSingleton;>//获取静态变量
 * 14 ifnonnull 27 (+13)//判空
 * 17 new #3 <DoubleCheckLockSingleton>//如果等于null,创建对象
 * 20 dup
 * 21 invokespecial #4 <DoubleCheckLockSingleton.<init> : ()V>//对象的初始化
 * 24 putstatic #2 <DoubleCheckLockSingleton.instance : LDoubleCheckLockSingleton;>//给变量赋值
 * 27 aload_0
 * 28 monitorexit
 */


高并发下,若全按照上面的顺序来执行就和我们期望的一样了,但是不然。编号21和24的顺序是不确定的,也就是存在半初始化问题,也就是24执行在前,21后。首先在锁机制下,当一条线程执行锁中的代码时,其他需要锁的线程全部就被挡在了锁的外部等待。该线程进入锁内,在对象没有真正的初始化时就对变量进行了赋值(此时对象里面的值都是堆中jvm给的默认值),外部的线程执行到 if (instance == null),发现不为空,立即返回该对象。那么线程就拿到了一个错误的对象


单例双重检查锁问题的解决


添加volatile


private static volatile DoubleCheckLockSingleton instance = null;


分析之前先补充知识:


volatile 的底层实现是通过插入内存屏障实现(C++,汇编实现)。
  • 每个 volatile 写操作前面插入一个 StoreStore 屏障
  • 每个 volatile 写操作后面插入一个 StoreLoad 屏障
  • 每个 volatile 读操作后面插入一个 LoadLoad 屏障
  • 每个 volatile 读操作后面插入一个 LoadStore 屏障


内存屏障


  • StoreStore 屏障可以保证在 volatile 写之前,其前面的所有普通写操作都已经刷新到主内存中
  • StoreLoad 屏障的作用是避免 volatile 写与后面可能有的 volatile 读/写操作重排序
  • LoadLoad 屏障用来禁止处理器把上面的 volatile 读与下面的普通读重排序
  • LoadStore 屏障用来禁止处理器把上面的 volatile 读与下面的普通写重排序


8375de4bacf146bab49c8301539e097d.png


JVM规定volatile需要实现的内存屏障


e0b8be4b63294336886bd8fbb21c52da.png


分析


首先在锁机制下,当一条线程执行锁中的代码时,其他需要锁的线程全部就被挡在了锁的外部等待。该线程进入锁内,执行到instance = new DoubleCheckLockSingleton(); **发现instance变量是被volatile修饰的,于是在该条语句的前后分别添加内存屏障。**防止了指令的重排。这样就解决了双锁检查单例的问题。





相关文章
|
Web App开发 监控 JavaScript
监控和分析 JavaScript 内存使用情况
【10月更文挑战第30天】通过使用上述的浏览器开发者工具、性能分析工具和内存泄漏检测工具,可以有效地监控和分析JavaScript内存使用情况,及时发现和解决内存泄漏、过度内存消耗等问题,从而提高JavaScript应用程序的性能和稳定性。在实际开发中,可以根据具体的需求和场景选择合适的工具和方法来进行内存监控和分析。
|
6月前
|
存储 弹性计算 缓存
阿里云服务器ECS经济型、通用算力、计算型、通用和内存型选购指南及使用场景分析
本文详细解析阿里云ECS服务器的经济型、通用算力型、计算型、通用型和内存型实例的区别及适用场景,涵盖性能特点、配置比例与实际应用,助你根据业务需求精准选型,提升资源利用率并降低成本。
473 3
|
2月前
|
安全 Java 数据库连接
一把锁的两种承诺:synchronized如何同时保证互斥与内存可见性?
临界区指多线程中访问共享资源的代码段,需通过互斥机制防止数据不一致与竞态条件。Java用`synchronized`实现同步,保证同一时刻仅一个线程执行临界区代码,并借助happens-before规则确保内存可见性与操作顺序,从而保障线程安全。
161 11
|
2月前
|
设计模式 缓存 Java
【JUC】(4)从JMM内存模型的角度来分析CAS并发性问题
本篇文章将从JMM内存模型的角度来分析CAS并发性问题; 内容包含:介绍JMM、CAS、balking犹豫模式、二次检查锁、指令重排问题
130 1
|
5月前
|
存储 人工智能 自然语言处理
AI代理内存消耗过大?9种优化策略对比分析
在AI代理系统中,多代理协作虽能提升整体准确性,但真正决定性能的关键因素之一是**内存管理**。随着对话深度和长度的增加,内存消耗呈指数级增长,主要源于历史上下文、工具调用记录、数据库查询结果等组件的持续积累。本文深入探讨了从基础到高级的九种内存优化技术,涵盖顺序存储、滑动窗口、摘要型内存、基于检索的系统、内存增强变换器、分层优化、图形化记忆网络、压缩整合策略以及类操作系统内存管理。通过统一框架下的代码实现与性能评估,分析了每种技术的适用场景与局限性,为构建高效、可扩展的AI代理系统提供了系统性的优化路径和技术参考。
297 4
AI代理内存消耗过大?9种优化策略对比分析
|
JavaScript
如何使用内存快照分析工具来分析Node.js应用的内存问题?
需要注意的是,不同的内存快照分析工具可能具有不同的功能和操作方式,在使用时需要根据具体工具的说明和特点进行灵活运用。
589 159
|
9月前
|
存储 Java
课时4:对象内存分析
接下来对对象实例化操作展开初步分析。在整个课程学习中,对象使用环节往往是最棘手的问题所在。
|
9月前
|
Java 编译器 Go
go的内存逃逸分析
内存逃逸分析是Go编译器在编译期间根据变量的类型和作用域,确定变量分配在堆上还是栈上的过程。如果变量需要分配在堆上,则称作内存逃逸。Go语言有自动内存管理(GC),开发者无需手动释放内存,但编译器需准确分配内存以优化性能。常见的内存逃逸场景包括返回局部变量的指针、使用`interface{}`动态类型、栈空间不足和闭包等。内存逃逸会影响性能,因为操作堆比栈慢,且增加GC压力。合理使用内存逃逸分析工具(如`-gcflags=-m`)有助于编写高效代码。
198 2
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
448 1
|
开发框架 监控 .NET
【Azure App Service】部署在App Service上的.NET应用内存消耗不能超过2GB的情况分析
x64 dotnet runtime is not installed on the app service by default. Since we had the app service running in x64, it was proxying the request to a 32 bit dotnet process which was throwing an OutOfMemoryException with requests >100MB. It worked on the IaaS servers because we had the x64 runtime install
252 5