当前位置: 首页 > news >正文

<多线程章节八> 单例模式中的饿汉模式与懒汉模式的讲解,以及懒汉模式中容易引起的Bug

💐专栏导读

本篇文章收录于多线程,也欢迎翻阅博主的其他文章,可能也会让你有不一样的收获😄
🌷JavaSE 🍂多线程 🌾数据结构

文章目录

  • 💐专栏导读
  • 💡饿汉模式
  • 💡懒汉模式
  • 💡懒汉模式多线程版
  • 💡volatile防止指令重排序

单例模式是一种经典的设计模式了,它的作用就是保证在有些场景下, 需要一个类只能有一个对象,而不能有多个对象,比如像你以后娶媳妇,你娶媳妇肯定是只能娶一个,而不能娶两个;

但是,问题来了,一个类只需要一个对象,那在new对象的时候只new一次对象不就可以了么,为什么还要弄个这么麻烦的东西呢?

因为啊,只new一次对象确实是只有一个,但是呢,如果你在写代码的过程中忘了呢,然后又new了一次,这种概率是很大的,毕竟,人是最不靠谱的动物😅,就像是有一句话说的好:宁可相信世界上有鬼,也不要相信男人的那张嘴😂,所以的,为了防止这种失误发生,就有了单例模式,在Java中也有许多类似的机制,比如final,就会保证修饰的变量肯定是不能改变的;@override,保证你这方法肯定是一个重写方法;**interface,**保证肯定要重写接口里面的方法;这些都是在语法方面进行了一些限制,但是,在语法方面,对于单例并没有特定的语法,所以,这里就通过编程技巧来达到类似的限制效果;

单例模式的两种实现方式:

💡饿汉模式

1.在类中实例化类的对象,给外界提供一个方法来使用这个对象;

2.将构造方法用private修饰,保证在类外不能再实例化这个类对象

public class SingleTon {//在类的内部实例化对象public static SingleTon instance = new SingleTon();//定义一个方法,用来获取这个对象//后序如果类外的代码想要使用对象时,直接调用这个方法即可public static SingleTon getInstance() {return instance;}//设置一个私有的构造方法,保证在这个类外无法实例化这个对象private SingleTon(){}
}

在这里插入图片描述

可以看到,这里的对象被static修饰,所以在类被加载的时候创建,创建的时机就比较早,并且被static修饰的对象只会被创建一次,所以这种在类加载时就创建实例的模式称为饿汉模式

💡懒汉模式

懒汉模式单线程版:

这样的写法与上面的相同点就是:同样在类外不能再第二次实例化对象,不同点是:将创建对象的时机放在getInstance方法中,这样在类加载的时候就不会创造实例,而是当第一次调用这个方法时才会去创建

public class SingleTon {public static SingleTon instance = null;//定义一个方法,用来获取这个对象//后序如果类外的代码想要使用对象时,直接调用这个方法即可public static SingleTon getInstance() {//懒汉模式if(instance == null) {instance = new SingleTon();}return instance;}//设置一个私有的构造方法,保证在这个类外无法实例化这个对象private SingleTon(){}
}

在这里插入图片描述

💡懒汉模式多线程版

在线程安全方面,上面的饿汉模式是线程安全的,而懒汉模式在多线程下是不安全的;

因为,如果多个线程同时访问一个变量,那么不会出现不安全问题,如果多个线程同时修改一个变量,就有可能出现不安全问题;

饿汉模式下,只进行了访问,没有涉及到修改

在这里插入图片描述

懒汉模式下,不仅进行了访问,还涉及了修改,那么下面就讲解以下懒汉模式在多线程下如何会产生不安全

在这里插入图片描述
在这里插入图片描述

既然出现了不安全问题,那么如何将懒汉模式修改成安全的呢?

💡方法:进行加锁,使线程安全

在这里插入图片描述

但是,如果锁加在这个地方,仍然是不安全的,因为,这样还是会进行穿插执行,并没有保证它是一个整体(f非原子性)

在这里插入图片描述

并不是加了锁就安全,只有锁加对了才会安全,在加锁的时候要保证以下几方面:

1.锁的 {} 的范围是合理的,能够把需要作为整体的每个部分都包括进去;

2.锁的对象能够起到锁竞争的效果;

懒汉模式多线程版改进👇

将if语句和new都放在锁里面称为一个整体,这样就避免了会穿插执行;

    public static SingleTon getInstance() {synchronized (SingleTon.class) {if(instance == null) {instance = new SingleTon();}}return instance;}

但是上述代码还有一个问题,每当调用getInstance时,都会尝试去进行加锁,而加锁是一个开销很大的操作,而这里的懒汉模式之所以会出现线程不安全问题,是因为只是在在第一次调用getInstance方法进行new对象时,可能会出现问题,但是,只要new完对象以后,就不用再进行锁竞争再去进行if判断了,直接访问就可以了,所以再次进行优化👇:

    public static SingleTon getInstance() {//在最外面在进行一次判断if(instance == null) {synchronized (SingleTon.class) {if(instance == null) {instance = new SingleTon();}}}return instance;}

在第一次实例化对象后,以后再调用个getInstance方法时,就不会再创建对象,而且也不会再去获取锁,因为,第一个if判断语句都不会进去,所以不会执行到加锁的语句;

上面的单例模式看着好像是完全没问题了,但是,还是有一个问题,就是可能会触发指令重排序问题,所以就需要使用volatile解决指令重排序问题

💡volatile防止指令重排序

指令重排序:编译器会保证在你代码逻辑不变的情况下,对代码进行优化,使代码的性能得到提高,这样的操作称为指令重排序;

举个例子:

在这里插入图片描述
在这里插入图片描述

在代码中,在实例化对象这一步可能会出现指令重排序问题,下面就来讲解一下为什么👇

在这里插入图片描述

在这里插入图片描述

对于上述的指令重排序问题,解决方案就是:使用volatile关键字修饰singleTon

**线程安全的单例模式(懒汉模式)**👇

public class SingleTon {//使用volatile关键字修饰,防止指令重排序public static volatile SingleTon singleTon = null;public static SingleTon getSingleTon() {if(singleTon == null) {synchronized (SingleTon.class) {if(singleTon == null) {singleTon = new SingleTon();}}}return singleTon;}private SingleTon() {};}

💡💡这里再次提醒,使用单例模式要注意三个要点:

  • 加锁
  • 两层if判断
  • 使用volatile修饰引用,防止指令重排序

相关文章:

  • 13.7性能测试工具(LoadRunner)(简单扫盲)
  • 【设计模式】第25节:行为型模式之“访问者模式”
  • 通过gosec白盒扫描Go代码中的SQL注入
  • Pytorch 猫狗识别案例
  • DAY40 343. 整数拆分 + 96. 不同的二叉搜索树
  • LeetCode 面试题 16.10. 生存人数
  • [USACO23OPEN] Field Day S题解
  • CCF_A 计算机视觉顶会CVPR2024投稿指南以及论文模板
  • Kamailio uac_replace和uac_restore
  • 电脑屏幕监控软件,能够帮助企业完成哪些事情?
  • C++ 单例模式
  • 学习redis之前的泛泛而谈(特性介绍,应用场景,Ubuntu安装与通用命令介绍)
  • KnowledgeGPT:利用检索和存储访问知识库上增强大型语言模型10.30
  • VMware虚拟网络连接的三种方式
  • 面试算法44:二叉树中每层的最大值
  • 07.Android之多媒体问题
  • Angular2开发踩坑系列-生产环境编译
  • AzureCon上微软宣布了哪些容器相关的重磅消息
  • If…else
  • javascript 哈希表
  • JSONP原理
  • MQ框架的比较
  • Webpack 4 学习01(基础配置)
  • 从@property说起(二)当我们写下@property (nonatomic, weak) id obj时,我们究竟写了什么...
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 聚簇索引和非聚簇索引
  • 深入 Nginx 之配置篇
  • 世界上最简单的无等待算法(getAndIncrement)
  • 小李飞刀:SQL题目刷起来!
  • 要让cordova项目适配iphoneX + ios11.4,总共要几步?三步
  • 智能网联汽车信息安全
  • puppet连载22:define用法
  • 国内唯一,阿里云入选全球区块链云服务报告,领先AWS、Google ...
  • 哈罗单车融资几十亿元,蚂蚁金服与春华资本加持 ...
  • ​​​​​​​ubuntu16.04 fastreid训练过程
  • ​如何防止网络攻击?
  • # Swust 12th acm 邀请赛# [ A ] A+B problem [题解]
  • #QT项目实战(天气预报)
  • (delphi11最新学习资料) Object Pascal 学习笔记---第2章第五节(日期和时间)
  • (LeetCode) T14. Longest Common Prefix
  • (附源码)springboot青少年公共卫生教育平台 毕业设计 643214
  • (更新)A股上市公司华证ESG评级得分稳健性校验ESG得分年均值中位数(2009-2023年.12)
  • (蓝桥杯每日一题)平方末尾及补充(常用的字符串函数功能)
  • (六)Hibernate的二级缓存
  • (深度全面解析)ChatGPT的重大更新给创业者带来了哪些红利机会
  • (十)T检验-第一部分
  • (十一)JAVA springboot ssm b2b2c多用户商城系统源码:服务网关Zuul高级篇
  • (四)七种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (转载)Google Chrome调试JS
  • .cn根服务器被攻击之后
  • .net CHARTING图表控件下载地址
  • .net core 调用c dll_用C++生成一个简单的DLL文件VS2008
  • .NET Framework .NET Core与 .NET 的区别
  • .net Signalr 使用笔记