多线程 双重检查锁详解

🍓 简介:java系列技术分享(👉持续更新中...🔥)

🍓 初衷:一起学习、一起进步、坚持不懈

🍓 如果文章内容有误与您的想法不一致,欢迎大家在评论区指正🙏

🍓 希望这篇文章对你有所帮助,欢迎点赞 👍 收藏 ⭐留言 📝

一、未加锁的单例

懒汉模式实现

java 复制代码
public class Singleton {
    
    private static Singleton instance;
    
    private Singleton() {
    }
    
    public Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }    
}

这是一个最简单的单例模式,在单线程下运转良好。但在多线程下会出现明显的问题,可能会创建多个实例。

可以看到,当两个线程同时执行时,是有可能会创建多个实例的,这很明显不符合单例的要求。

二、加锁单例

java 复制代码
public class Singleton {
    
    private static Singleton instance;
    
    private Singleton() {
    }
    
    public synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

与第一个示例唯一的区别是在方法上添加了synchronized关键字。这时,当多个线程进入该方法时,需要先获得锁才能进行执行。

  1. 通过在方法上添加synchronized关键字,看似完美的解决了多线程的问题,但却带了性能问题

  2. 我们知道使用锁会导致额外的性能开销,对于上面的单例模式,只有第一次创建时需要锁(防止创建多个实例),但查询时是不需要锁的

  3. 如果针对方法进行加锁,每次查询也要承担加锁的性能损耗

三、双重检查锁

java 复制代码
public class Singleton {
    
    private static Singleton instance;
    
    private Singleton() {
    }
    
    public Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}
  1. 缩小锁的范围;

  2. 锁之前先判断一下是不是null,如果不为null,说明已经实例化了,直接返回,没必要进行创建;

  3. 如果为null,进行加锁,然后再次判断是否为null。为什么要再次判断?因为一个线程判断为null之后,另外一个线程可能已经创建了对象,所以在锁定之后,需要再次核实一下,真的为null,则进行对象创建。

改进之后,既保证了线程的安全性,又避免了锁导致的性能损失。问题到此结束了吗?并没有,继续往下看

四、JVM的指令重排

在JVM当中,编译器为了性能问题,会进行指令重排。

在上述代码中new Singleton()并不是原子操作,有可能会被编译器进行重排操作。

当线程A执行完步骤赋值操作,但还未执行对象初始化。此时,线程B进来了,在第一层判断时发现Instance已经有值了(实际上还未初始化),直接返回对应的值。那么,程序在使用这个未初始化的值时,便会出现错误。

针对此问题,可在instance上添加volatile关键字,使得instance在读、写操作前后都会插入内存屏障,避免重排序。

最终,单例模式实现如下:

java 复制代码
public class Singleton {
    
    private static volatile Singleton instance;
    
    private Singleton() {
    }
    
    public Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

五、总结

  • 未加锁单例模式使用,会创建多个对象
  • 方法上加锁,导致性能下降
  • 代码内局部加锁,双重判断,既满足线程安全,又满足性能需求
  • 单例模式特例:创建对象分多步,会出现指令重排现象,采用volatile进行避免指令重排;
相关推荐
疯狂的程序猴3 分钟前
完整指南:iPhone崩溃日志查看与分析方法及低内存崩溃处理
后端
秧歌star51913 分钟前
PageHelper 分页失效原因分析与正确实践
后端
沐怡旸15 分钟前
【底层机制】Android OTA更新系统:原理与应用深度解析
android·面试
疯狂的程序猴18 分钟前
苹果iOS应用签名与上架App Store完整指南包括注意事项
后端
回家路上绕了弯18 分钟前
生产环境服务器变慢?从应急到根因的全流程诊断处理指南
分布式·后端
小胖霞18 分钟前
Node+Express+MySQL 后端生产环境部署,实现注册功能(三)
前端·后端
一 乐19 分钟前
农产品电商|基于SprinBoot+vue的农产品电商系统(源码+数据库+文档)
java·前端·javascript·数据库·vue.js·spring boot
aiopencode25 分钟前
抓包技术全面指南:原理、工具与应用场景
后端
烤麻辣烫44 分钟前
23种设计模式(新手)-7迪米特原则 合成复用原则
java·开发语言·学习·设计模式·intellij-idea
该用户已不存在1 小时前
Gemini 3.0 发布,Antigravity 掀桌,程序员何去何从?
后端·ai编程·gemini