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

7.多线程之单例模式

单例模式

文章目录

  • 单例模式
  • 1. 什么是单例模式
  • 2. 饿汉模式
  • 3. 懒汉模式
    • 3.1 单线程版:
    • 3.2 多线程版

1. 什么是单例模式

    单例模式是一种设计模式,常见的设计模式还有工厂模式、建造者模式等。
设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。我们使用这些设计模式,相当于是前辈们在一些特殊的场景下总结出的经验,我们直接使用即可,相当于站在巨人的肩膀上编程!!!
    单例模式是针对一些特定的类,只允许创建一个实例。为什么会有单例模式?因为引入单例模式后,在代码层面会有强制检查,不会允许创建出第二个实例,避免一些bug,人为检查是及其不可靠、不可控的。
    单例模式有两种实现方式,分别是饿汉模式和懒汉模式,我们就这两种实现展开。

2. 饿汉模式


class Singleton{private static Singleton instance = new Singleton();public static Singleton getInstance() {return instance;}private Singleton() {}
}
public class Test {public static void main(String[] args) {Singleton sgt1 = Singleton.getInstance();Singleton sgt2 = Singleton.getInstance();System.out.println(sgt1 == sgt2);}
}    

运行结果~~

true进程已结束,退出代码为 0

    饿汉模式首先是线程安全的,对象在类加载时就会创建好,然后将构造方法设为私有,不允许再创建实例,最后通过公有方法将这个对象提供给外层调用。
    这种模式为什么叫饿汉模式呢?其实也很简单,无论外层需不需要使用这个对象,在类加载时这个对象就已经创建了。而相比于饿汉模式,懒汉模式的做法则更加高效,在第一次实用类的时候,在进行创建。

3. 懒汉模式

3.1 单线程版:

class Singleton {private static Singleton instance = null;private Singleton() {}public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

    为什么说这样实现懒汉模式是单线程版本?因为这个版本的懒汉模式在并发环境下,会产生线程安全问题,即线程不安全。
    线程安全问题发生在首次创建实例时. 如果在多个线程中同时调getInstance 方法, 就可能导致创建出多个实例。一旦实例已经创建好了, 后面再多线程环境调用 getInstance 就不再有线程安全问题了(不再修改instance 了),因此枷锁(synchronized)可以一定程度上改善这个问题。

在这里插入图片描述
    这个版本的懒汉模式,在多线程环境下,一但有其他线程在第一次创建对象时穿插(OS系统调度)进行执行,就可能会创建出多个实例。

3.2 多线程版

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

    在这个版本中对if判断进行枷锁,保证了创建对象时不会有其它线程穿插执行,在进一步给instance使用volatile修饰(保证可见性、顺序),从而保证了线程安全。我们进一步发现,只有在第一次进入时,才会需要创建对象,而如果每次都枷锁,就会降低程序的执行效率,所以我么在外部又嵌套了一层if。此处要注意的是这两层if虽然条件一样,但是含义却不一样。
    锁竞争(枷锁解锁)的开销是比较大的,而懒汉模式只有在第一次才需要创建对象(产生锁竞争),后续就不在需要锁竞争了,也不必枷锁。

  1. 外层if :外层if是判断单例模式的对象是否创建出来,如果已经创建,则直接返回instance(instance被volatile修饰,不会产生内存可见性问题),不会产生锁竞争。
  2. 内层if:当线程第一次执行时会进入内层,竞争同一把锁,假设线程1拿到锁,就可以创建出对象,此时线程1执行完后,由于没内存可见性问题,其它线程就会发现instance已经不为空了,从而不会继续进入外层if竞争锁对象,降低了锁竞争的开销。

    如果本篇文章对你有帮助,请点赞、评论、转发,你的支持是我创作的动力!!!

http://www.lryc.cn/news/210619.html

相关文章:

  • docker 常用指令
  • Numpy、Pandas、Matplotlib学习(更新ing...)
  • Linux-sdio接口
  • 【MATLAB源码-第62期】基于蜣螂优化算法(DBO)的无人机三维地图路径规划,输出最短路径和适应度曲线。
  • RealVNC Enterprise 7.7.0 Crack
  • Windows下Jenkins自动化部署SpringBoot应用
  • Leetcode59螺旋矩阵2
  • 抖音很火的情侣飞行棋/真心话大冒险/情侣扫雷html网站源码带在线支付、代理分销等!
  • Word批量删除文档属性和个人信息方法图解
  • Kubernetes(K8s)从入门到精通系列之十八:使用 Operator Lifecycle Manager(OLM) 安装operator
  • request、response请求转发和重定向
  • MySql学习路线
  • Vue实现首页导航和左侧菜单,介绍mock.js并实现登录注册间的跳转,实现左侧栏折叠效果,优化Main.vue组件,使用mock.js生成随机响应数据
  • 汇总下之RobotFramework自动化框架的系列文章
  • 0052【Edabit ★☆☆☆☆☆】Learn Lodash: _.drop, Drop the First Elements of an Array
  • 从无人机到实景三维海洋系统
  • 第二章前端开发ES6基础
  • 使用Python将PDF转为图片
  • JAVA 同城服务货运搬家小程序系统开发时应注意哪些?
  • uniapp开发小程序 小米手机真机bottom:0无效 底部间隙 设备安全区域处理办法
  • 四十、【进阶】索引失效情况2
  • 敏感词过滤--golang
  • NVIDIA cuda安装时全部失败
  • APScheduler-调度器AsyncIOScheduler
  • 二十三种设计模式全面解析-原型模式(Prototype Pattern)详解:创造对象的奇妙之道
  • el-input 给icon图标绑定点击事件
  • 计算机网络——物理层
  • 常用排序算法
  • MGRE环境下的OSPF
  • 【计算机毕设】微信小程序案例-学生签到系统