爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      首页 知识中心 软件开发 文章详情页

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      2023-05-11 06:07:50 阅读次数:133

      单例模式

      (目录)


      单例模式

      什么是单例模式

      要求我们代码中的某个类,只能有一个实例,不能有多个实例。 实例就是对象。 就是说某个类只能new 一个对象,不能new多个对象。

      这种单例模式,在实际开发中是非常常见的,也是非常有用的。 开发中的很多“概念”,天然就是单例的。

      比如:

      ​ 使用JDBC操作数据库,此时数据库连接可以通过数据库连接池数据库连来获取

          DataSource ds=new MySqlDataSource();
      
          Connection c=ds.getConnection() ;//获取数据库连接
      

      连接池就可以使用单例模式,创建唯一的一个对象.

      大部分跟数据有关的东西,服务器里面只存一份。那么,就都可以使用“单例模式”来进行表示。


      单例模式的两种经典实现

      单例模式中有两个典型实现:

      1. 饿汉模式
      2. 懒汉模式

      我们来通过一个生活上的例子来给大家讲讲什么是饿汉模式,什么是懒汉模式。  

      洗碗:

      1. 第一种情况: 假设我们中午吃饭的时候,一家人用了4个碗。然后吃完之后,马上就把碗给洗了 这种情况,就是饿汉模式

      2. 第二种情况: 中午吃饭的时候,一家人用了4个碗。然后吃完之后,碗先放着,不着急洗 等待晚上吃饭的时候,发现只需要2个碗。 那么就将 4个没洗的碗 中,洗出2个碗,拿来用。吃完之后,碗先放着,不着急洗。 如果下一顿只用一个玩,就洗出1个碗 就是用多少,拿多少。少的不够,多的不要 这就是懒汉模式

      但是在计算机中,普遍认为 懒汉模式 比 饿汉模式好。 主要因为 懒汉模式 的效率更高

      也很好理解:洗 2 个 碗,肯定比洗4个碗轻松。

      所以用几个洗几个。 根据需要,进行操作。

      “懒” 这个字一般 在计算机中,是一个褒义词。


      1. 饿汉模式 (线程安全)

      1. 饿汉模式的类在类被加载的过程中就会立刻实例化一个对象 所以后续无论如何操作 只要严格使用get方法 就不会出现其他的实例
      2. 由上可知 饿汉模式是线程安全的
      3. 他也会有一个问题就是即使我不使用这个类 他还会创建一个实例来占用我们的内存 这就导致他的效率就不高
      4. 这个版本的单个实例不能有其他实例变量, 不然还是会出现非线性安全问题 (非线程安全问题就是有个线程对用一个实例变量修改造成数据的脏读)
      //饿汉模式
      class Singleton{
          // 1、使用 static 创建一个实例,并且立即进行实例化,
          private  static  Singleton instance = new Singleton();
          // 2、为了防止程序员在其他地方不小心new这个 Singleton,需要把这个类的构造方法设置为 private
          private Singleton(){};
          //3、提供一个方法,让外面能够拿到唯一的实例。
          public static Singleton getInstance(){
              return instance;
          }
      
      }
      

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      分析:

      (1) 为什么是线程安全的?

      ​ 由于在类加载的时候就初始化了(第2行),只有一份,所以是线程安全的.

      (2) 缺点

      ①还没有使用,就浪费了内存空间.

      ②new对象(没有执行类加载,会先执行,再执行成员变量+实例代码块+构造方法),可能抛异常,也就是还没有调用getInstance,在类加载时就抛出了异常,那么以后调用getInstance方法时,都是会出现问题的.


      2. 懒汉模式 (线程不安全)

      1. 懒汉模式就改进了饿汉模式的缺点 他只有在使用的时候才会让该类去实例化一个对象 并且此后再去获取对象只能获取这一个对象
      2. 所以我们一般认为懒汉模式比饿汉模式效率更高
      3. 但是懒汉模式也有缺点他线程不安全 这点在可以优化解决

      由于比较懒,你让我干活,我才开始,否则我不会主动干的~

      ​ 说人话就是: 类加载时,不急着初始化对象,第一次调用,初始化对象,后边再次调用,直接返回第一次创建好的对象.

      //单例模式 - 懒汉模式
      class Singleton2{
          //1、现在就不是立即初始化实例
          private static Singleton2 instance;// 默认值:Null
          //2、把构造方法设为 private
          private Singleton2(){};
          //3、提供一个公开的方法,来获取这个 单例模式的唯一实例
          public static Singleton2 getInstance(){
              // 只有当我们真正用到这个实例的时候,才会真正去创建这个实例
              if(instance == null){
                  instance = new Singleton2();
              }
              return instance;
          }
      }
      
      public class Test20 {
          public static void main(String[] args) {
              Singleton2 instance = Singleton2.getInstance();
          }
      }
      

      分析:

      为什么是线程不安全的?

      在getInstance()方法中,总共有三行Java代码。

      而且其中instance = new Singleton() 会被分解成三行字节码指令,相当于并发并行的对多行共享变量的操作,所以是线程不安全的.

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)


      区别:

      饿汉模式 和 懒汉模式 的唯一区别:

      在于 创建实例的时机不一样。

      饿汉模式 是 类加载时,创建。 懒汉模式 是 首次使用时,创建。

      所以懒汉模式就更懒一些,不用的时候,不创建;等到用用的时候,再去创建。 这样做的目的,就是节省资源。


      其实在计算机很多其它场景中,也会涉及这情况。

      一个典型的案例:

      notepad 这样的程序(记事本软件),在打开大文件的时候是很慢的。 假如,你要打开一个 1G 大小的文件,此时 notepad 就会尝试把这 1 G 的 所有内容都读到内存中。 将 1G 的数据量 存入 内存,显然是非常慢的。 不管你要不要,全部都给你。 这就是 饿汉模式。

      问题也随之而来:这些数据,我们真的能全部用得到吗?显示是不太可能的。 因此就会浪费很多资源。   像一些其他的程序,在打开大文件的时候就有优化。 假设也是打开 1G的文件,但是只先加载这一个屏幕中能显示出来的部分。 看到哪,加载到哪里。这样不会用空间上的浪费 这就是 懒汉模式


      实现线程安全的单例模式

      1. 懒汉模式+synchronized静态同步方法 (线程安全但效率差)

      说到让一个代码线程安全,我们自然而然的就想到加锁! 但是问题就在于:在哪个地方加锁合适呢? 其实也很好观察,将 if 语句的执行操作 给 加锁,使其两个操作为原子性。 直白来说: 就是 if 语句 打包成“一个整体”,就跟前面分析 count++ 一样。 一致性执行完。

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      加锁范围 一定要包含 if 语句!!! 要不然没有效果,就像下面这样! 【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      本来我们是想将 读 和 写 操作,打包成一个整体, 但是现在只是 针对写操作进行加锁,这时候就跟没加锁 一样,是没有区别的。

      请大家注意!并不是代码中有 synchronized,一定就是线程安全的。 这需要看 synchronized 加的位置,也要正确。 所以 synchronized 写的位置。不能随便。   回过头来,我们再来看一下 synchronized 锁的对象写我们应该些什么。

      //单例模式 - 懒汉模式
      class Singleton2{
          //1、就不是立即初始化实例
          private static volatile Singleton2  instance;// 默认值:Null
          //2、把构造方法设为 private
          private Singleton2(){};
          //3、提供一个公开的方法,来获取这个 单例模式的唯一实例
          public static Singleton2 getInstance(){
                  // 只有当我们真正用到这个实例的时候,才会真正去创建这个实例
                  synchronized(Singleton2.class){
                      if(instance == null){
                          instance = new Singleton2();
                      }
                  }
              return instance;
          }
      }
      

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      虽然我们确实通过上述加锁操作,解决了 if 语句 的原子性问题。

      但是!这样的程序,还存在这几个问题!

      1.代码执行效率问题

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      2、指令重排序 虽然其他线程再调用 单例线程的时候,也是加了 synchronized 的。 减缓了循环速度,从而保证了 内存可见性。 但是!还有一个问题,来看下面。

      【多线程】 实现单例模式 ( 饿汉、懒汉 ) 实现线程安全的单例模式 (双重效验锁)

      此时,我们才完成一个线程安全的单例模式 - 懒汉模式 1、正确的位置加锁 2、双重if判定 3、volatile关键字


      2. 懒汉模式+二次判断(双重校验锁,线程安全且效率高)

      用法:

      • 双重校验: 两个if.
      • 锁: synchronized
      • 注意使用静态变量(引用) 使用volatile.
      class Singleton {
          private static volatile Singleton instance = null;
          private Singleton() {}
          public static Singleton getInstance() { 
              if (instance == null) {  //第5行
                  synchronized (Singleton.class) { //第6行
                      if (instance == null) {  //第7行
                          instance = new Singleton(); //第8行
                      } 
              }
              return instance; //第9行
          }
      }
      

      分析:

      • ①对象初始化前,多线程调用getInstance(),需要保证线程安全,即执行第5,6,7,8行.
      • ②对象初始化以后,只执行 if 判断和 return 语句,即只执行第5,9行.

      ​ ② 明显比 ① 执行的情况多很多,所以考虑不加锁,提高效率.

      ​ ② 不需要加锁,可以使用volatile保证可见性.

      即:

      ​ 第5行: 初始化完成之后,不需要加锁,使用volatile修饰变量,保证可见性,能满足线程安全,代码行本身就是原子性. 可以并行并发的执行,提高了效率.

      ​ 第6行: 没有初始化完成时,创建对象需要加锁来保证线程安全.

      ​ 第7行: 竞争锁失败的线程,还会执行同步代码块,需要再次判断,保证只初始化一次.

      ​ 第9行: 引用使用了volatile关键字,还有建立内存屏障,禁止指令重排序的功能(new 分解的三条指令: 分配内存空间,初始化对象,赋值给变量)


      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://blog.51cto.com/panyujie/5434370,作者:Perceus,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:【Java Web】引入markdown编辑页面

      下一篇:9,公有云实战——高并发访问时流量分发和会话保持的实现

      相关文章

      2024-06-07 07:32:50

      Java设计模式之单例模式

      在某些情况下,需要确保一个类只有一个实例,并且需要一个全局访问点来访问这个实例。

      2024-06-07 07:32:50
      Java , 单例模式
      2024-06-06 08:03:43

      【设计模式】单例模式——反序列化破坏单例模式的情景

      一个单例对象创建好之后,有时候需要将对象序列化然后进行数据持久化,下次使用的时候通过反序列化转化为内存对象。反序列化后的对象会重新分配内存,会破坏单例模式。

      2024-06-06 08:03:43
      单例模式
      2024-06-06 08:03:43

      【设计模式】单例模式——单例模式变体之“多例模式”

      所谓“多例模式”并不在GoF的23种设计模式之内,是单例模式中的一种特例,在很多资料中也被称为单例模式的容器式实现。

      2024-06-06 08:03:43
      单例模式
      2024-06-05 09:49:00

      【设计模式】单例模式——单例模式的懒汉式和DCL式实现

      【设计模式】单例模式——单例模式的懒汉式和DCL式实现

      2024-06-05 09:49:00
      加锁 , 单例模式
      2024-06-05 09:49:00

      【设计模式】单例模式——其他框架源码中的单例模式

      单例模式在很多开源框架和第三方服务中都有应用场景,本文只讨论最流行的框架。

      2024-06-05 09:49:00
      单例 , 单例模式
      2024-06-05 09:49:00

      【设计模式】单例模式——单例模式在其他框架源码中的应用

      单例模式在其他框架源码中也有广泛的应用。

      2024-06-05 09:49:00
      单例模式
      2024-06-05 09:09:23

      【设计模式】单例模式——单例模式的饿汉式和枚举式实现

      【设计模式】单例模式——单例模式的饿汉式和枚举式实现

      2024-06-05 09:09:23
      单例模式 , 设计模式
      2024-06-03 09:10:05

      单例模式

      单例模式

      2024-06-03 09:10:05
      单例模式
      2024-05-29 09:01:43

      设计模式之单例设计模式

      单例模式(Singleton Pattern)属于创建型模式,它提供了一种创建对象的最佳方式。

      2024-05-29 09:01:43
      单例模式 , 设计模式
      2024-04-25 09:04:29

      【设计模式-2.1】创建型——单例模式

      设计模式根据用途分为创建型、结构性和行为型。创建型模式主要用于描述如何创建对象,本文介绍创建型中的单例模式。

      2024-04-25 09:04:29
      java , 单例模式 , 设计模式
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5224821

      查看更多

      最新文章

      Java设计模式之单例模式

      2024-06-07 07:32:50

      程序员必知!单例模式的实战应用与案例分析

      2024-04-19 06:40:32

      BeanDefinition属性

      2024-04-15 09:06:20

      Thread专题(14) - 自定义Annocation

      2024-03-28 08:10:08

      设计模式面试题-单例模式【JavaPub版】

      2024-03-22 08:13:30

      Python 设计模式:单例模式

      2023-04-27 08:01:21

      查看更多

      热门文章

      Python 设计模式:单例模式

      2023-04-27 08:01:21

      Java线程安全问题与同步锁

      2023-03-01 02:38:47

      Thread专题(14) - 自定义Annocation

      2024-03-28 08:10:08

      设计模式面试题-单例模式【JavaPub版】

      2024-03-22 08:13:30

      BeanDefinition属性

      2024-04-15 09:06:20

      Java设计模式之单例模式

      2024-06-07 07:32:50

      查看更多

      热门标签

      java Java python 编程开发 代码 开发语言 算法 线程 Python html 数组 C++ 元素 javascript c++
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      Thread专题(14) - 自定义Annocation

      Python 设计模式:单例模式

      设计模式面试题-单例模式【JavaPub版】

      Java线程安全问题与同步锁

      程序员必知!单例模式的实战应用与案例分析

      Java设计模式之单例模式

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号