单例模式(Singleton)

单例模式保证一个类仅有一个实例,并提供一个全局访问点来访问它,这个类称为单例类。可见,在实现单例模式时,除了保证一个类只能创建一个实例外,还需提供一个全局访问点。

Singleton is a creational design pattern that lets you ensure that a class has only one instance, 
while providing a global access point to this instance.  

为提供一个全局访问点,可以使用全局变量,但全局变量无法禁止用户实例化多个对象。为此,可以让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问实例的方法。
综上,单例模式的要点有三个:一是某个类只能有一个实例;二是这个类必须自行创建这个实例;三是这个类必须自行向整个系统提供这个实例。

结构设计

单例模式只有一个角色:
Singleton,单例类,用来保证实例唯一并提供一个全局访问点。为实现访问点全局唯一,可以定义一个静态字段,同时为了封装对该静态字段的访问,可以定义一个静态方法。为了保证实例唯一,这个类还需要在内部保证实例的唯一。基于以上思考,单例模式的类图表示如下:
请添加图片描述

伪代码实现

接下来将使用代码介绍下单例模式的实现。单例模式的实现方式有很多种,主要的实现方式有以下五种:饿汉方式、懒汉方式、线程安全实现方式、双重校验方式、惰性加载方式。

(1) 饿汉方式

饿汉方式就是在类加载的时候就创建实例,因为是在类加载的时候创建实例,所以实例必唯一。由于在类加载的时候创建实例,如果实例较复杂,会延长类加载的时间。

// 1. 定义单例类,提供全局唯一访问点,保证实例唯一
public class HungrySingleton {// (1) 声明并实例化静态私有成员变量(在类加载的时候创建静态实例)private static final HungrySingleton instance = new HungrySingleton();// (2) 私有构造方法private HungrySingleton() {}// (3) 定义静态方法,提供全局唯一访问点public static HungrySingleton getInstance() {return instance;}public void foo() {System.out.println("---------do some thing in a HungrySingleton instance---------");}
}
// 2. 客户端调用
public class HungrySingletonClient {public void test() {// (1) 获取实例HungrySingleton singleton = HungrySingleton.getInstance();// (2) 调用实例方法singleton.foo();}
}

(2) 懒汉方式

懒汉方式就是在调用实例获取(如getInstance())接口时,再创建实例,这种方式可避免在加载类的时候就初始化实例。

// 1. 定义单例类,提供全局唯一访问点,保证实例唯一
public class LazySingleton {// (1) 声明静态私有成员变量private static LazySingleton instance;// (2) 私有构造方法private LazySingleton() {}// (3) 定义静态方法,提供全局唯一访问点public static LazySingleton getInstance() {// 将实例的创建延迟到第一次获取实例if(instance == null) {instance = new LazySingleton();}return instance;}public void foo() {System.out.println("---------do some thing in a LazySingleton instance---------");}
}
// 2. 客户端调用
public class LazySingletonClient {public void test() {// (1) 获取实例LazySingleton instance = LazySingleton.getInstance();// (2) 调用实例方法instance.foo();}
}

需要说明的是,对多线程语言来说(如java语言),懒汉方式会带来线程不安全问题。如果在实例前执行判空处理时,至少两个线程同时进入这行代码,则会创建多个实例。
所以,对于多线程语言来说,为了保证代码的正确性,还需在实例化的时候,保证线程安全。

(3) 线程安全实现方式

为保证线程安全,可以在实例判空前,进行线程同步处理,如添加互斥锁。

// 1. 定义单例类,提供全局唯一访问点,保证实例唯一
public class ThreadSafeSingleton {// (1) 声明静态私有成员变量private static ThreadSafeSingleton instance;// (2) 私有构造方法private ThreadSafeSingleton() {}// (3) 定义静态方法,提供全局唯一访问点public static ThreadSafeSingleton getInstance() {// 使用synchronized方法,保证线程安全synchronized (ThreadSafeSingleton.class) {if (Objects.isNull(instance)) {instance = new ThreadSafeSingleton();}return instance;}}public void foo() {System.out.println("---------do some thing in a ThreadSafeSingleton instance---------");}
}
// 2. 客户端调用
public class ThreadSafeSingletonClient {public void test() {// (1) 获取实例ThreadSafeSingleton instance = ThreadSafeSingleton.getInstance();// (2) 调用实例方法instance.foo();}
}

但是这种方式,会因线程同步而带来性能问题。因为大多数场景下,是不存在并发访问。

(4) 双重校验方式

为避免每次创建实例时加锁带来的性能问题,引入双重校验方式,即在加锁前额外进行实例判空校验,这样就可保证非并发场景下仅在第一次实例化时,去加锁并创建实例。

// 1. 定义单例类,提供全局唯一访问点,保证实例唯一
public class DoubleCheckSingleton {// (1) 声明静态私有成员变量private static volatile DoubleCheckSingleton instance;// (2) 私有构造方法private DoubleCheckSingleton() {}// (3) 定义静态方法,提供全局唯一访问点public static DoubleCheckSingleton getInstance() {// 在加锁之前,先执行判空检验,提高性能if (Objects.isNull(instance)) {// 使用synchronized方法,保证线程安全synchronized (DoubleCheckSingleton.class) {if (Objects.isNull(instance)) {instance = new DoubleCheckSingleton();}}}return instance;}public void foo() {System.out.println("---------do some thing in a DoubleCheckSingleton instance---------");}
}
// 2. 客户端调用
public class DoubleCheckSingletonClient {public void test() {// (1) 获取实例DoubleCheckSingleton instance = DoubleCheckSingleton.getInstance();// (2) 调用实例方法instance.foo();}
}

注意,使用双重校验方式时,需明确语言是否支持指令重排序。以Java语言为例,实例化一个对象的过程是非原子的。具体来说,可以分为以下三步:(1) 分配对象内存空间;(2)将对象信息写入上述内存空间;(3) 创建对上述内存空间的引用。其中(2)和(3)的顺序不要求固定(无先后顺序),所以存在实例以分配内存空间但还未初始化的情况。如果此时存在并发线程使用了该未初始化的对象,则会导致代码异常。为避免指令重排序,Java语言中可以使用 volatile 禁用指令重排序。更多细节可以参考java单例模式一文。

(5) 惰性加载方式

由于加锁会带来性能损耗,最好的办法还是期望实现一种无锁的设计,且又能实现延迟加载。对Java语言来说,静态内部类会延迟加载(对C#语言来说,内部类会延迟加载)。可以利用这一特性,实现单例。

// 1. 定义单例类,提供全局唯一访问点,保证实例唯一
public class LazyLoadingSingleton {// (2) 私有构造方法private LazyLoadingSingleton() {}// (3) 定义静态方法,提供全局唯一访问点public static LazyLoadingSingleton getInstance() {// 第一调用静态类成员或方法时,才加载静态内部类,实现了延迟加载return Holder.instance;}public void foo() {System.out.println("---------do some thing in a LazyLoadingSingleton instance---------");}// (1) 声明私有静态内部类,并提供私有成员变量private static class Holder {private static LazyLoadingSingleton instance = new LazyLoadingSingleton();}
}
// 2. 客户端调用
public class LazyLoadingSingletonClient {public void test() {// (1) 获取实例LazyLoadingSingleton instance = LazyLoadingSingleton.getInstance();// (2) 调用实例方法instance.foo();}
}

很多框架代码都会引入静态内部类,实现延迟加载。更多静态内部类的使用细节可以参考笔者之前的文章。

适用场景

在以下情况下可以使用单例模式:
(1) 如果系统只需要一个实例对象,则可以考虑使用单例模式。
如提供一个唯一的序列号生成器,或者需要考虑资源消耗太大而只允许创建一个对象。
(2) 如果需要调用的实例只允许使用一个公共访问点,则可以考虑使用单例模式。
(3) 如果一个系统只需要指定数量的实例对象,则可以考虑扩展单例模式。
可以在单例模式中,通过限制实例数量实现多例模式。

优缺点

单例模式模式有以下优点:
(1) 提供了对唯一实例的受控访问。
因为单例类封装了它的唯一实例,所以它可以严格控制客户怎样以及何时访问它。
(2) 节约系统资源。由于在系统内存中只存在一个对象,因此可以节约系统资源,对于一些需要频繁创建和销毁的对象,单例模式无疑可以提高系统的性能。
(3) 允许可变数目的实例。可以基于单例模式进行扩展,使用与单例控制相似的方法来获得指定个数的对象实例。
但是单例模式模式也存在以下缺点:
(1) 违反了单一职责原则。单例类的职责过重,既充当工厂角色,提供了工厂方法,同时又充当产品角色,包含一些业务方法,将产品的创建产品本身的功能融合到一起,在一定程度上违背了单一职责原则。
(2) 单例类扩展困难。由于单例模式中没有抽象层,且继承困难,所以单例类的扩展有很大的困难。
(3) 滥用单例模式带来一些负面问题,如过多的创建单例,会导致这些单例类一直无法释放且占用内存空间,另外对于一些不频繁使用的但占用内存空间较大的对象,也不宜将其创建为单例。而且现在很多面向对象语言(如Java、C#)都提供了自动垃圾回收的技术。

参考

《设计模式:可复用面向对象软件的基础》 Erich Gamma, Richard Helm, Ralph Johnson, John Vlissides 著 李英军, 马晓星 等译
https://design-patterns.readthedocs.io/zh_CN/latest/creational_patterns/singleton.html 单例模式
https://refactoringguru.cn/design-patterns/singleton 单例模式
https://www.runoob.com/design-pattern/singleton-pattern.html 单例模式
https://www.cnblogs.com/adamjwh/p/9033554.html 单例模式
https://blog.csdn.net/czqqqqq/article/details/80451880 单例模式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/74102.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JavaScript场景应用:Canvas实战开发一个二维折线图插件

🏆作者简介,黑夜开发者,全栈领域新星创作者✌,阿里云社区专家博主,2023年6月csdn上海赛道top4。 🏆数年电商行业从业经验,历任核心研发工程师,项目技术负责人。 🏆本文已…

VB6中FSO具体应用详解

文前申明:原文为通用版实例代码,本菜鸟在每例之后加入一个简单的实例(均验证通过),供有需要的朋友参考. 您正在看的VB教程是:VB入门基础认识VB的文件系统对象FSO。 在 VB 编程中经常需要和文件系统打交道,比如获取硬盘的剩余空间、判断文件夹或文件是否存在等。在…

认识主被动无人机遥感数据、预处理无人机遥感数据、定量估算农林植被关键性状、期刊论文插图精细制作与Appdesigner应用开发

目录 第一章、认识主被动无人机遥感数据 第二章、预处理无人机遥感数据 第三章、定量估算农林植被关键性状 第四章、期刊论文插图精细制作与Appdesigner应用开发 更多推荐 遥感技术作为一种空间大数据手段,能够从多时、多维、多地等角度,获取大量的…

PHP语言基础知识(超详细)

文章目录 前言第一章 PHP语言学习介绍 1.1 PHP部署安装环境1.2 PHP代码工具选择 第二章 PHP代码基本语法 2.1 PHP函数知识介绍2.2 PHP常量变量介绍 2.2.1 PHP变量知识:2.2.2 PHP常量知识: 2.3 PHP注释信息介绍2.4 PHP数据类型介绍 2.4.1 整形数据类型2.4…

基于量子同态的安全多方量子求和加密

摘要安全多方计算在经典密码学中一直扮演着重要的角色。量子同态加密(QHE)可以在不解密的情况下对加密数据进行计算。目前,大多数协议使用半诚实的第三方(TP)来保护参与者的秘密。我们使用量子同态加密方案代替TP来保护各方的隐私。在量子同态加密的基础上&#xff…

2023年自动化测试已成为标配?一篇彻底打通自动化测试...

目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 首先我们从招聘岗…

《面试1v1》ElasticSearch 和 Lucene

🍅 作者简介:王哥,CSDN2022博客总榜Top100🏆、博客专家💪 🍅 技术交流:定期更新Java硬核干货,不定期送书活动 🍅 王哥多年工作总结:Java学习路线总结&#xf…

智慧~经典开源项目数字孪生智慧商场——开源工程及源码

深圳南山某商场的工程和源码免费赠送,助您打造智慧商场。立即获取,提升商场管理效能! 项目介绍 凤凰商场作为南山地区的繁华商业中心,提供多样化的购物和娱乐体验。通过此项目,凤凰商场将迈向更智能的商业模式。 本项目…

基于SaaS模式的Java基层卫生健康云HIS系统源码【运维管理+运营管理+综合监管】

云HIS综合管理平台 一、模板管理 模板分为两种:病历模板和报表模板。模板管理是运营管理的核心组成部分,是基层卫生健康云中各医疗机构定制电子病历和报表的地方,各医疗机构可根据自身特点特色定制电子病历和报表,制作的电子病历…

Python-Python基础综合案例:数据可视化 - 折线图可视化

版本说明 当前版本号[20230729]。 版本修改说明20230729初版 目录 文章目录 版本说明目录知识总览图Python基础综合案例:数据可视化 - 折线图可视化json数据格式什么是jsonjson有什么用json格式数据转化Python数据和Json数据的相互转化 pyecharts模块介绍概况如何…

Golang 函数参数的传递方式 值传递,引用传递

基本介绍 我们在讲解函数注意事项和使用细节时,已经讲过值类型和引用类型了,这里我们再系统总结一下,因为这是重难点,值类型参数默认就是值传递,而引用类型参数默认就是引用传递。 两种传递方式(函数默认都…

BUG分析以及BUG定位

一般来说bug大多数存在于3个模块: 1、前台界面,包括界面的显示,兼容性,数据提交的判断,页面的跳转等等,这些bug基本都是一眼可见的,不太需要定位,当然也不排除一些特殊情况&#xf…

《cuda c编程权威指南》04 - 使用块和线程索引映射矩阵索引

目录 1. 解决的问题 2. 分析 3. 方法 4. 代码示例 1. 解决的问题 利用块和线程索引,从全局内存中访问指定的数据。 2. 分析 通常情况下,矩阵是用行优先的方法在全局内存中线性存储的。如下。 8列6行矩阵(nx,ny)(…

Kafka-消费者组消费流程

消费者向kafka集群发送消费请求,消费者客户端默认每次从kafka集群拉取50M数据,放到缓冲队列中,消费者从缓冲队列中每次拉取500条数据进行消费。

时序预测 | Python实现NARX-DNN空气质量预测

时序预测 | Python实现NARX-DNN空气质量预测 目录 时序预测 | Python实现NARX-DNN空气质量预测效果一览基本介绍研究内容程序设计参考资料效果一览 基本介绍 时序预测 | Python实现NARX-DNN空气质量预测 研究内容 Python实现NARX-DNN空气质量预测,使用深度神经网络对比利时空气…

PDF文件忘记密码,怎么办?

PDF文件设置密码分为打开密码和限制密码,忘记了密码分别如何解密PDF密码? 如果是限制编辑密码忘记了,我们可以试着将PDF文件转换成其他格式来避开限制编辑,然后重新将文件转换回PDF格式就可以了。 如果因为转换之后导致文件格式…

如何打造属于自己的个人IP?

在当今信息爆炸的时代,个人 IP 已经成为人们在网络世界中的独特标签。无论是在职场上、创业中,还是在社交生活中,拥有个人 IP 的人都能脱颖而出,吸引更多的关注和机会。那么,如何打造属于自己的个人 IP 呢?…

go 如何知道一个对象是分配在栈上还是堆上?

如何判断变量是分配在栈(stack)上还是堆(heap)上? Go和C不同,Go局部变量会进行逃逸分析。如果变量离开作用域后没有被引用,则优先分配到栈上,否则分配到堆上。判断语句:…

在外远程NAS群晖Drive - 群晖Drive挂载电脑磁盘同步备份【无需公网IP】

文章目录 前言1.群晖Synology Drive套件的安装1.1 安装Synology Drive套件1.2 设置Synology Drive套件1.3 局域网内电脑测试和使用 2.使用cpolar远程访问内网Synology Drive2.1 Cpolar云端设置2.2 Cpolar本地设置2.3 测试和使用 3. 结语 前言 群晖作为专业的数据存储中心&…

如何开启一个java微服务工程

安装idea IDEA常用配置和插件(包括导入导出) https://blog.csdn.net/qq_38586496/article/details/109382560安装配置maven 导入source创建项目 修改项目编码utf-8 File->Settings->Editor->File Encodings 修改项目的jdk maven import引入…