单例模式的优缺点和使用场景

2023-11-12

单例模式(Singleton),也叫单子模式,是一种常用的软件设计模式。在应用这个模式时,单例对象的类必须保证只有一个实例存在。许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息。这种方式简化了在复杂环境下的配置管理。 

实现单例模式的思路是: 
    一个类能返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名 称);当我们调用这个方法时,如果类持有的引用不为空就返回这个引用,如果类保持的引用为空就创建该类的实例并将实例的引用赋予该类保持的引用;同时我们 还将该类的构造函数定义为私有方法,这样其他处的代码就无法通过调用该类的构造函数来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例。 

需要注意的地方: 
    单例模式在多线程的 应用场合下必须小心使用。如果当唯一实例尚未创建时,有两个线程同时调用创建方法,那么它们同时没有检测到唯一实例的存在,从而同时各自创建了一个实例, 这样就有两个实例被构造出来,从而违反了单例模式中实例唯一的原则。 解决这个问题的办法是为指示类是否已经实例化的变量提供一个互斥锁(虽然这样会降低效率)。 

优点: 
    1.在单例模式中,活动的单例只有一个实例,对单例类的所有实例化得到的都是相同的一个实例。这样就 防止其它对象对自己的实例化,确保所有的对象都访问一个实例 
    2.单例模式具有一定的伸缩性,类自己来控制实例化进程,类就在改变实例化进程上有相应的伸缩性。 
    3.提供了对唯一实例的受控访问。 
    4.由于在系统内存中只存在一个对象,因此可以 节约系统资源,当 需要频繁创建和销毁的对象时单例模式无疑可以提高系统的性能。 
    5.允许可变数目的实例。 
    6.避免对共享资源的多重占用。 
缺点: 
    1.不适用于变化的对象,如果同一类型的对象总是要在不同的用例场景发生变化,单例就会引起数据的错误,不能保存彼此的状态。 
    2.由于单利模式中没有抽象层,因此单例类的扩展有很大的困难。 
    3.单例类的职责过重,在一定程度上违背了“单一职责原则”。 
    4.滥用单例将带来一些负面问题,如为了节省资源将数据库连接池对象设计为的单例类,可能会导致共享连接池对象的程序过多而出现连接池溢出;如果实例化的对象长时间不被利用,系统会认为是垃圾而被回收,这将导致对象状态的丢失。 
使用注意事项: 
    1.使用时不能用反射模式创建单例,否则会实例化一个新的对象 
    2.使用懒单例模式时注意线程安全问题 
    3.饿单例模式和懒单例模式构造方法都是私有的,因而是不能被继承的,有些单例模式可以被继承(如登记式模式) 
适用场景: 
    单例模式只允许创建一个对象,因此节省内存,加快对象访问速度,因此对象需要被公用的场合适合使用,如多个模块使用同一个数据源连接对象等等。如: 
    1.需要频繁实例化然后销毁的对象。 
    2.创建对象时耗时过多或者耗资源过多,但又经常用到的对象。 
    3.有状态的工具类对象。 
    4.频繁访问数据库或文件的对象。 
以下都是单例模式的经典使用场景: 
    1.资源共享的情况下,避免由于资源操作时导致的性能或损耗等。如上述中的日志文件,应用配置。 
    2.控制资源的情况下,方便资源之间的互相通信。如线程池等。 
应用场景举例: 
    1.外部资源:每台计算机有若干个打印机,但只能有一个PrinterSpooler,以避免两个打印作业同时输出到打印机。内部资源:大多数软件都有一个(或多个)属性文件存放系统配置,这样的系统应该有一个对象管理这些属性文件 
    2. Windows的Task Manager(任务管理器)就是很典型的单例模式(这个很熟悉吧),想想看,是不是呢,你能打开两个windows task manager吗? 不信你自己试试看哦~ 
    3. windows的Recycle Bin(回收站)也是典型的单例应用。在整个系统运行过程中,回收站一直维护着仅有的一个实例。 
    4. 网站的计数器,一般也是采用单例模式实现,否则难以同步。 
    5. 应用程序的日志应用,一般都何用单例模式实现,这一般是由于共享的日志文件一直处于打开状态,因为只能有一个实例去操作,否则内容不好追加。 
    6. Web应用的配置对象的读取,一般也应用单例模式,这个是由于配置文件是共享的资源。 
    7. 数据库连接池的设计一般也是采用单例模式,因为数据库连接是一种数据库资源。数据库软件系统中使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的效率损耗,这种效率上的损耗还是非常昂贵的,因为何用单例模式来维护,就可以大大降低这种损耗。 
    8. 多线程的线程池的设计一般也是采用单例模式,这是由于线程池要方便对池中的线程进行控制。 
    9. 操作系统的文件系统,也是大的单例模式实现的具体例子,一个操作系统只能有一个文件系统。 
    10. HttpApplication 也是单位例的典型应用。熟悉ASP.Net(IIS)的整个请求生命周期的人应该知道HttpApplication也是单例模式,所有的HttpModule都共享一个HttpApplication实例. 
   
实现单利模式的原则和过程: 
    1.单例模式:确保一个类只有一个实例,自行实例化并向系统提供这个实例 
    2.单例模式分类:饿单例模式(类加载时实例化一个对象给自己的引用),懒单例模式(调用取得实例的方法如getInstance时才会实例化对象)(java中饿单例模式性能优于懒单例模式,c++中一般使用懒单例模式) 
    3.单例模式要素: 
        a.私有构造方法 
        b.私有静态引用指向自己实例 
        c.以自己实例为返回值的公有静态方法 

1.饿汉式:单例实例在类装载时就构建,急切初始化。(预先加载法)

  /**
  * 饿汉式(推荐)
  *
  */
  public class Test {
          private Test() {
          }
          public static Test instance = new Test();
          public Test getInstance() {
                 return instance;
         }
 }

优点 
    1.线程安全 
    2.在类加载的同时已经创建好一个静态对象,调用时反应速度快 
缺点 
    资源效率不高,可能getInstance()永远不会执行到,但执行该类的其他静态方法或者加载了该类(class.forName),那么这个实例仍然初始化 


2.懒汉式:单例实例在第一次被使用时构建,延迟初始化。

  class Test {
          private Test() {
          }
          public static Test instance = null;
          public static Test getInstance() {
                  if (instance == null) {
                //多个线程判断instance都为null时,在执行new操作时多线程会出现重复情况
                          instance = new Singleton2();
                  }
                 return instance;
         }
 }

优点: 
    避免了饿汉式的那种在没有用到的情况下创建事例,资源利用率高,不执行getInstance()就不会被实例,可以执行该类的其他静态方法。 
缺点: 
    懒汉式在单个线程中没有问题,但多个线程同事访问的时候就可能同事创建多个实例,而且这多个实例不是同一个对象,虽然后面创建的实例会覆盖先创建的实例,但是还是会存在拿到不同对象的情况。解决这个问题的办法就是加锁synchonized,第一次加载时不够快,多线程使用不必要的同步开销大。 

3.双重检测 

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

优点 
    资源利用率高,不执行getInstance()就不被实例,可以执行该类其他静态方法 
缺点 
    第一次加载时反应不快,由于java内存模型一些原因偶尔失败 


4.静态内部类 

class Test {
          private Test() {
          }
  
          private static class SingletonHelp {
                  static Test instance = new Test();
          }
  
          public static Test getInstance() {
                 return SingletonHelp.instance;
         }
 }

优点 
    资源利用率高,不执行getInstance()不被实例,可以执行该类其他静态方法 
缺点 
    第一次加载时反应不够快 

总结: 
    一般采用饿汉式,若对资源十分在意可以采用静态内部类,不建议采用懒汉式及双重检测

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

单例模式的优缺点和使用场景 的相关文章

随机推荐

  • 【ESP32-CAM】使用opencv获取ESP32-CAM视频流,并将图像保存至TF卡(一)

    VSCode python opencv ESP32 CAM 本项目仅作为学习记录 不定时更新 Arduino 对于ESP32 CAM 我们使用Arduino来开发 首先需要准备一些硬件 ESP32 CAM 在淘宝大约30rmb一个 烧录底
  • 回归预测算法比较

    GBDT和时间序列分析的区别 GBDT算法做预测分析时考虑到一些影响目标值的因素 而时间序列主要考虑到数据在时序上的一些规律 忽略了其他因素对目标值的影响 这两种算法在实际应用中刚好互补 如果原始数据中包含足够的数据特征 通常情况下使用GB
  • FaceForensics和FaceForensics++【参考文献30和31】

    30 FaceForensics学术界最大的合成视频数据库之一 介绍了一个新的面部操作数据集 大约有50万张经过编辑的图像 来自1000多个视频 这些操作是用最先进的面部编辑方法生成的 它超过了所有现有的视频操作数据集至少一个数量级 利用我
  • 今天,大语言模型革新百度搜索

    机器之心原创 作者 泽南 别搜关键词了 百度搜索直接给你正确答案 最近几天 一些最常用的工具正在被大厂重新发明 ChatGPT 横空出世后 整个科技领域突然就开上了快车道 今年 2 月 微软推出的 AI 版必应 用最先进的大语言模型重构了搜
  • C语言之指针的概念必备练习题

    对于指针 想必是多少人刚开始学习时候的梦魇 笔者也一样 一套关于指针概念的练习题请大家多多仔细阅读 体会不一样的感觉 1 作业标题 711 关于指针的概念 错误的是 作业内容 A 指针是变量 用来存放地址 B 指针变量中存的有效地址可以唯一
  • 用MOS管驱动电机吧

    图示为我应用于实际的空心杯电机驱动电路 可见 该电路通过MCU的PWM信号控制电机 实现了无刷电机的无极调速 该电路的基本思想就是通过PWM信号控制MOS管 SI2302 的开与关 改变PWM的占空比 进而改变流过电机的有效电流 从而控制电
  • python thread_Python thread模块用法详解

    该包在 Python 2 中属于正常可用状态 但在 Python 3 中处于即将废弃的状态 虽然还可以用 但包名被改为 thread 使用 thread 包首先要引入该包 在 Python 2 中使用下面的语句来引入 import thre
  • 关于Python脚本开头两行的:#!/usr/bin/python和# -*- coding: utf-8 -*-的作用 – 指定

    usr bin python 是用来说明脚本语言是python的 是要用 usr bin下面的程序 工具 python 这个解释器 来解释python脚本 来运行python脚本的 coding utf 8 是用来指定文件编码为utf 8的
  • 大数据24小时:京东启动“人工智能加速器”项目,网传王劲因“家庭原因”离开景驰

    数据猿导读 王劲因 家庭原因 离开景驰 原CTO韩旭接任CEO一职 加强区块链布局 京东启动 人工智能加速器 项目 美国云服务商GTT收购欧洲最大云平台营运商 交易金额达23亿美元 以下为您奉上更多大数据热点事件 编辑 abby 官网 ww
  • Eclipse上Maven环境配置使用 (全)

    Eclipse上Maven环境配置使用 全 1 安装配置Maven 1 1 从Apache网站 http maven apache org 下载并且解压缩安装Apache Maven Maven下载地址 http maven apache
  • 30个适合女生玩的可爱网站

    ugmbbc发布于 2008 03 20 13 30 12 2905 次阅读 字体 大 小 打印预览 感谢不要笑我的投递这次推荐给大家的都是非常好玩和可爱的网站 他们都拥有不错的技术和创意 这些网站尤其适合女孩子玩 当然cnBeta是一个罗
  • 河南省计算机考试题目,计算机二级考试模拟题型

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 21 若不想修改数据库文件中的数据库对象 打开数据库文件时要选择 B A 以独占方式打开 B 以只读方式打开 C 以共享方式打开 D 打开 22 某文本型字段的值只能为字母且长度为6 则可将该字
  • 深度学习的一些教程

    几个不错的深度学习教程 基本都有视频和演讲稿 附两篇综述文章和一副漫画 还有一些以后补充 Jeff Dean 2013 Stanford http i stanford edu infoseminar dean pdf 一个对DL能干什么的
  • web自动点击遍历(入口检查)

    目录 web自动点击遍历 入口检查 一 背景 二 技术方案 三 核心代码 3 1代码结构 3 2用例模块 3 3获取HTML代码 3 4正则查找a标签 3 5遍历存储 3 6递归遍历 web自动点击遍历 入口检查 导语 web自动化的存在问
  • Qt debug版本运行正常release版本运行崩溃问题记录

    问题由来 某一项目debug版本运行正常 进入发布阶段 结果release后的版本出现了崩溃问题 因为是release版本 不能debug运行 只能通过printf debug 虽然问题原因很简单 但是耗费了大量时间 故做此记录 以供后续参
  • 网络安全(黑客)自学笔记&学习路线

    谈起黑客 可能各位都会想到 盗号 其实不尽然 黑客是一群喜爱研究技术的群体 在黑客圈中 一般分为三大圈 娱乐圈 技术圈 职业圈 娱乐圈 主要是初中生和高中生较多 玩网恋 人气 空间 建站收徒玩赚钱 技术高的也是有的 只是很少见 技术圈 这个
  • Jeesite4本地及服务器上传文件、图片详解

    大家过年好 你们的老朋友小Q又回来了 最近一段时间忙于公司项目开发 又加上过年比较忙 所以没能留出时间更新内容 大家是不是有点着急了 公司最近在使用jeesite4开发项目 我呢 对这个框架一点也不熟悉 遇到问题都是现查现用 这不 在上传文
  • 【Vue3源码学习】响应式 api: watchEffect

    在 Vue2中watch 是 option 写法中一个很常用的选项 使用它可以非常方便的监听一个数据源的变化 而在 Vue3 中watch 独立成了一个 响应式api 源码地址 packages runtime core src apiWa
  • IOS发送通知与接收通知

    1 注册通知 注册通知 NSNotificationCenter defaultCenter addObserver self selector selector notiReceived name RefreshOrder object
  • 单例模式的优缺点和使用场景

    单例模式 Singleton 也叫单子模式 是一种常用的软件设计模式 在应用这个模式时 单例对象的类必须保证只有一个实例存在 许多时候整个系统只需要拥有一个的全局对象 这样有利于我们协调系统整体的行为 比如在某个服务器程序中 该服务器的配置