软件设计模式系列之六——单例模式
1 模式的定义
单例模式(Singleton Pattern)是一种常见的创建型设计模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点来获取该实例。这意味着无论何时何地,只要需要该类的实例,都会返回同一个实例,而不是创建多个相同的实例。单例模式通常用于管理全局状态、资源共享或限制某些资源的访问。
2 举例说明
在日常生活中,随处可见单例模式的例子,比如你家中有一台电视,通常只需要一个遥控器来控制它。无论家里谁想看电视,都会使用同一个遥控器,而且遥控器只能让家里人轮流使用,也就是不能有多个人同时使用遥控器控制电视。这个遥控器就是一个单例,因为它确保只有一个实例存在,并且提供了一个全局的访问点,以便你可以随时使用它。
3 结构
单例模式的结构包括以下要素:
-
单例类(Singleton Class):单例模式的核心是单例类,它负责管理唯一的实例。通常,这个类会将其构造函数设为私有,以防止外部直接实例化多个对象。单例类会定义一个静态方法或变量来获取或创建唯一的实例。
-
私有构造函数(Private Constructor):单例类的构造函数通常会被设置为私有,这样外部无法直接实例化这个类。私有构造函数的目的是确保只有单例类内部可以创建类的实例。
-
静态成员变量(Static Member Variable):单例类会包含一个私有的静态成员变量,用于保存唯一的实例。这个成员变量通常被命名为 instance 或类似的名称。
-
静态方法(Static Method):单例类会提供一个公共的静态方法,通常命名为 getInstance() 或类似的名称,用于获取或创建唯一的实例。这个方法会检查是否已经存在实例,如果存在则返回现有实例,否则创建一个新的实例并返回它。
单例模式的关键是将构造函数私有化,以确保只有一个实例,并提供一个全局的方法来获取这个实例,以实现全局唯一性。这种结构确保了在应用程序中只有一个实例存在,无论何时何地都可以访问这个实例,从而实现了单例模式的设计目标。
4 实现步骤
实现单例模式的关键步骤通常包括以下几个:
-
将构造函数私有化(Private Constructor):在单例模式中,首先需要将单例类的构造函数设为私有,以防止外部直接实例化多个对象。这是确保只有一个实例的重要步骤。
-
创建一个私有的静态成员变量(Private Static Member Variable):单例类内部通常会包含一个私有的静态成员变量,用于保存唯一的实例。这个变量通常被命名为 instance 或类似的名称。
-
提供一个公共的静态方法(Public Static Method):单例类会提供一个公共的静态方法,通常命名为 getInstance() 或类似的名称,用于获取或创建唯一的实例。这个方法会检查是否已经存在实例,如果存在则返回现有实例,否则创建一个新的实例并返回它。
-
在获取实例时进行实例化(Lazy Initialization):在 getInstance() 方法中,需要检查 instance 是否为 None,如果为 None,则创建一个新的实例并将其赋值给 instance,否则直接返回 instance。这确保了实例在需要时才会被创建,避免了不必要的开销。
-
处理多线程环境(Thread Safety):如果应用程序可能在多线程环境下使用单例类,需要考虑线程安全性。可以使用加锁机制来确保在多线程环境下也只有一个实例被创建。
5 代码实现
在Java中,可以使用懒汉式和饿汉式两种方式来实现单例模式。下面分别给出这两种方式的示例代码:
懒汉式单例模式
在懒汉式中,实例是在首次被请求时才创建。
public class LazySingleton {private static LazySingleton instance;private LazySingleton() {// 私有构造函数,防止外部实例化}public static LazySingleton getInstance() {if (instance == null) {instance = new LazySingleton();}return instance;}
}
在懒汉式中,getInstance 方法首先检查实例是否已经创建。如果没有创建实例,则创建一个新的实例并返回。这种实现延迟了实例的创建,只有在需要时才会创建。
饿汉式单例模式
在饿汉式中,实例在类加载时就被创建,无论是否需要。
public class EagerSingleton {private static final EagerSingleton instance = new EagerSingleton();private EagerSingleton() {// 私有构造函数,防止外部实例化}public static EagerSingleton getInstance() {return instance;}
}
在饿汉式中,实例在类加载时就被创建,因此无论何时需要实例,都可以立即返回。这种实现简单且线程安全,但可能会造成资源浪费,因为实例会在应用程序启动时就被创建。
需要注意的是,懒汉式在多线程环境下需要额外的同步措施来确保线程安全,而饿汉式天生是线程安全的。选择使用哪种方式取决于具体的需求和性能考虑。
6 典型应用场景
单例模式在各种应用场景中都有广泛的应用,主要用于确保一个类只有一个实例,并提供全局访问点。以下是一些常见的单例模式应用场景:
数据库连接池:在大多数应用程序中,与数据库的交互是常见的操作。为了提高性能和资源利用率,应用程序通常会使用数据库连接池来管理数据库连接。单例模式可以用于确保只有一个数据库连接池的实例存在,以避免多次创建和销毁数据库连接。
线程池:线程池用于管理和控制线程的执行。通过使用单例模式,可以确保只有一个线程池实例,从而更有效地管理并发执行的任务。
配置管理:在应用程序中,通常需要读取和管理配置信息,例如数据库连接参数、应用程序设置等。单例模式可用于存储和管理这些配置数据,以确保在整个应用程序中使用相同的配置。
日志记录器:在应用程序中记录日志是一项重要的任务,通常会使用日志记录器来处理日志信息。通过单例模式,可以确保只有一个日志记录器实例,以避免多次初始化和配置日志记录器。
窗口管理器:在图形用户界面应用程序中,窗口管理器用于管理应用程序窗口的创建、销毁和切换。单例模式可用于确保只有一个窗口管理器实例,以维护窗口状态和顺序。
单例模式在需要确保全局唯一性、资源共享、全局访问和状态管理的各种应用场景中非常有用。它可以帮助简化代码、提高性能,并确保应用程序的一致性。然而,需要谨慎使用,以避免引入全局状态和多线程问题。
7 优缺点
优点:
全局访问点:通过单例模式,可以在应用程序的任何地方轻松访问相同的实例。
资源共享:单例模式可用于管理共享的资源,例如数据库连接、线程池等,以提高性能和资源利用率。
避免重复创建:单例模式确保只有一个实例,避免了重复创建对象的开销。
缺点:
可能引入全局状态:过度使用单例模式可能导致全局状态,使得代码难以维护和测试。
不适用于多线程环境:如果不正确地实现单例模式,可能会导致多线程竞态条件,需要额外的同步机制来解决。
8 类似模式
在软件开发中,单例模式和原型模式通常在创建和管理"bean"(也称为对象或组件)时发挥重要作用,但它们在此上下文中有不同的用途和应用场景。
单例模式在bean的创建中的应用:
Spring框架中的单例bean:在Spring框架中,默认情况下,Spring容器会将Bean配置为单例(Singleton)。这意味着每个bean在应用程序中只有一个实例,并且Spring容器负责管理这些单例bean的生命周期。这种单例模式的应用确保了全局唯一性,并且可以节省资源和提高性能。
原型模式在bean的创建中的应用:
原型范围的Spring bean:在Spring框架中,你可以将bean配置为原型(Prototype)范围,这意味着每次从Spring容器请求该bean时,都会创建一个新的实例。原型模式的应用适用于那些需要频繁创建新实例的场景,例如HTTP请求的处理,每个请求需要一个新的bean实例以避免状态共享。
关系和应用场景:
单例模式通常用于那些需要确保全局唯一性的bean,例如服务层的单例组件、数据库连接池、配置管理器等。它适用于那些需要共享状态或资源的情况。
原型模式通常用于那些需要频繁创建新实例的bean,例如Web应用程序中的请求处理器、线程池中的任务、HTTP会话管理器等。它适用于那些需要隔离状态或资源的情况。
在Spring框架中,你可以根据bean的具体需求将它们配置为单例或原型范围,以满足应用程序的不同要求。这两种模式有各自的优势和适用场景,可以根据业务逻辑和性能要求来选择合适的范围。
9 小结
单例模式是一种有用的设计模式,用于确保一个类只有一个实例,并提供全局访问点。它在多种应用场景中都有用武之地,但需要小心使用,以避免引入全局状态和多线程问题。通过将构造函数私有化、使用静态变量保存实例以及提供一个静态方法来获取实例,可以实现单例模式。在设计应用程序时,要考虑是否需要使用单例模式来满足特定的需求。
相关文章:
软件设计模式系列之六——单例模式
1 模式的定义 单例模式(Singleton Pattern)是一种常见的创建型设计模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点来获取该实例。这意味着无论何时何地,只要需要该类的实例,都会返回同一个…...
verdi dump状态机的波形时直接显示状态名
前段时间看到别人用verdi看状态机的波形时,可以显示定义的状态参数,觉得很有意思,特地学习了一下 通常拉出状态机信号的波形是下面这样的 这种信号,我们要想知道每个数值代表的状态,还需要跟定义的parameter比对 像这…...
代码随想录算法训练营19期第53天
1143.最长公共子序列 视频讲解:动态规划子序列问题经典题目 | LeetCode:1143.最长公共子序列_哔哩哔哩_bilibili 代码随想录 初步思路:动态规划。 总结: dp[i][j] :长度为[0, i - 1]的字符串A与长度为[0, j - 1]…...
二刷力扣--栈和队列
栈和队列 栈和队列基础(Python) 栈一种先进后出,队列先进后出。 Python中可以用list实现栈,用append()模拟入栈,用pop()模拟出栈。 也可以用list实现队列,但是效率较低,一般用collections.deq…...
第六章 图 十、关键路径
开始顶点(源点): 在AOE网中仅有一个入度为0的顶点,称为开始顶点(源点),它表示整个工程的开始; 结束顶点(汇点): 也仅有一个出度为0的顶点,称为结束顶点(汇点)…...
Virtualbox固定存储硬盘转换为动态存储硬盘
现象 一开始分配固定存储过大,占了太多空间,现在想换成动态存储释放空闲空间。 解决 关闭虚拟机进入虚拟介质管理从使用的硬盘复制出一个动态存储硬盘在设置中把硬盘替换为副本硬盘 详细步骤参考: https://blog.csdn.net/qq_24033983/arti…...
【栈与队列面试题】有效的括号(动图演示)
leetcode20.括号匹配问题 前言: 💥🎈个人主页:Dream_Chaser~ 🎈💥 ✨✨刷题专栏:http://t.csdn.cn/UlvTc ⛳⛳本篇内容:力扣上栈与队列的面试OJ题目 目录 leetcode20.括号匹配问题 1.问题描…...
基于matlab实现的弹簧振动系统模型程序(动态模型)
完整代码: clear all; %System data m1.0; zeta0.01; omega01.0; Dt1.0; f01.0; x00.0; dotx00.0; xmaxsqrt(x0^2(dotx0/omega0)^2)min([0.5*abs(f0)*Dt/(m*omega0) f0/omega0^2]); omegadomega0*sqrt(1-zeta^2); dt00.1*pi/omega0; nstep500; a0.70; b0.…...
哨兵1号(Sentinel-1)SAR卫星介绍
1. 哥白尼计划 说起欧空局的哨兵1号,就不得不先说一下欧空局的“哥白尼计划”。 欧空局的哥白尼计划(Copernicus Programme)是欧空局与欧盟合作的一项极其重要的地球观测计划。该计划旨在提供免费开放的、可持续的地球观测数据,…...
[maven] scopes 管理 profile 测试覆盖率
[maven] scopes & 管理 & profile & 测试覆盖率 这里将一些其他的特性和测试覆盖率(主要是 jacoco) scopes maven 的 scope 主要就是用来限制和管理依赖的传递性,简单的说就是,每一个 scope 都有其对应的特性&…...
css网页打印字体设置
media print {font-family:"SimHei";color: #000;border-color: #000; }常用字符编码表 中文名英文名Unicode 编码黑体SimHeiSimHei微软雅黑Microsoft YaHei5FAE\8F6F\96C5\9ED1宋体SimSun\5B8B\4F53仿宋FangSong\4EFF\5B8B html5常用转义字符℃ 字符十…...
JAVA高级技术入门(单元测试,反射,注解,动态代理)
JAVA高级技术入门(单元测试,反射,注解,动态代理) 一、Junit单元测试二、反射1.认识反射,获取类概念:快速入门:获取Class对象的三种方式 2.1获取类的构造器2.2获取类的构造器的作用&a…...
uni-app 实现自定义按 A~Z 排序的通讯录(字母索引导航)
创建 convertPinyin.js 文件 convertPinyin.js 将下面的内容复制粘贴到其中 const pinyin (function() {let Pinyin function(ops) {this.initialize(ops);},options {checkPolyphone: false,charcase: "default"};Pinyin.fn Pinyin.prototype {init: functi…...
C++ PrimerPlus 复习 第一章 命令编译链接文件 make文件
第一章 命令编译链接文件 C 有什么呢?C 源代码文件后缀运行C过程可执行代码:编译语法:makeMakefile 基础语法编写完make只要和将要编译的文件放一起就行 然后在该目录使用make命令,就将自动运行;基础的Makefile版本 现…...
微信小程序——常用组件的属性介绍
常用的组件内容标签 text 文本组件类似于HTML中的span标签,是一个行内元素rich-text 富文本标签支持把HTML字符串渲染为WXML结构 text标签的基本使用 通过text组件的selectable属性,实现长按选中文本内容的效果。只有text标签支持长按选中效果&#x…...
【深度学习】 Python 和 NumPy 系列教程(廿七):Matplotlib详解:3、多子图和布局:散点矩阵图(Scatter Matrix Plot)
目录 一、前言 二、实验环境 三、Matplotlib详解 1、2d绘图类型 2、3d绘图类型 3、多子图和布局 1. subplot()函数 2. subplots()函数 3. 散点矩阵图(Scatter Matrix Plot) 一、前言 Python是一种高级编程语言,由Guido van Rossum于…...
解决jupyter打开的默认路径问题
已经安装完anaconda,但是jupyter每一次打开的路径都不是自己想要的路径,可以在配置文件中修改jupyter打开的默认路径,具体步骤如下: 首先打开anaconda的命令行 如果有多个环境的,需要输入conda activate 环境名称以下命…...
Git 学习笔记
Git 学习笔记 Git 简介 Git 是一个 开源的分布式版本控制系统。 什么是版本控制? 版本控制是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统。 什么是分布式版本控制系统? 介绍分布式版本控制系统前,有…...
【Qt】QGroundControl入门3:源码初探
1、源码目录 QGroundControl使用pro来管理工程,可以使用qmake来编译。同时还有CMakeLists.txt,应该可以使用cmake来编译,本人还没有尝试。 QGroundControl是跨平台的,支持android、win、linux、mac、ios系统,在QGCCommon.pri中可见关于跨平台编译的配置。 1.1 目录树 …...
腾讯mini项目-【指标监控服务重构】2023-07-31
今日已办 trace_id传播 关于如何使用 trace_id 创建 span 的思路 【暂未实现 & 测试】 调研 SpanProcessor 阅读源码的test 明日待办 根据 trace_id 创建 span,应该需要 parent span_id 才能有 trace 的树状 span 的关系...
Rust通用编程概念(3)
Rust通用编程概念 1.变量和可变性1.执行cargo run2.变量3.变量的可变性4.常量5.遮蔽5.1遮蔽与mut区别1.遮蔽2.mut 2.数据类型1.标量类型1.1整数类型1.2浮点数类型1.3数字运算1.4布尔类型1.5字符类型 2.复合类型2.1元组类型2.2数组类型1.访问数组2.无效的数组元素访问 3.函数3.1…...
学Python的漫画漫步进阶 -- 第四步
学Python的漫画漫步进阶 -- 第四步 四、运算符4.1 算术运算符4.2 比较运算符4.3 逻辑运算符4.4 位运算符4.5 赋值运算符4.6 运算符的优先级4.7 练一练4.8 运算符的总结全部16步完成后 ,后续就是介绍项目实战,请大家给予点赞、关注! 四、运算符…...
【LeetCode-中等题】18. 四数之和
文章目录 题目方法一:双指针(定2动2) 题目 方法一:双指针(定2动2) 这题可以参考【LeetCode-中等题】15. 三数之和 区别在于,三数之和只需要用一个for循环定住一个数,然后设置两个前…...
每日一题 102二叉树的层序遍历
题目 给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。 示例 1: 输入:root [3,9,20,null,null,15,7] 输出:[[3],[9,20],[15,7]] 示例 2:…...
牛客: BM4 合并两个排序的链表
牛客: BM4 合并两个排序的链表 文章目录 牛客: BM4 合并两个排序的链表题目描述题解思路题解代码 题目描述 题解思路 以链表一为主链表,遍历两条链表 若当前链表二的节点val小于当前链表一的下一个节点val,则将链表链表二的该节点连到链表一的节点的下一个,链表一的当前节点往…...
C语言基础知识点(六)二维数组指针和地址
#include <stdio.h>int main() {int a[2][3] {2, 4, 6,8, 10, 12};printf("a:%p, a1:%p\n", a, a 1); // 相差3*sizeof(int)12,二维数组名是一个指向每一行的指针,a:0061FF08, a1:0061FF14prin…...
nodejs格式化输入
需求 比如我现在要格式为Axxx-xxx(xxx是数字)的格式,但是输入有可能为A1-2这种情况,就需要补零,变成A001-002 代码实现 const regex /^A(\d)\-(\d)$/; // 正则匹配桩号合法格式const match input.match(regex);if…...
国家网络安全周 | 金融日,一起 get金融行业数据安全
2023国家网络安全宣传周 热度一直在持续! 9月15日是国家网络安全宣传金融日。 目前随着国际形势愈发严峻,金融机构基础设施的全面数字化升级,带来了全新的安全问题。数据安全不单是技术问题,更是已经成为一个关系社会稳定发展的…...
分布式事务解决方案之TCC
分布式事务解决方案之TCC 什么是TCC事务 TCC是Try、Confirm、Cancel三个词语的缩写,TCC要求每个分支事务实现三个操作:预处理Try、确认 Confirm、撤销Cancel。Try操作做业务检查及资源预留,Confirm做业务确认操作,Cancel实现一个…...
Git 的基础命令 码云 gitee
就比如,我们的开发吧,我自己本地分支是dqh,远程分支也是new //我开始提交代码 //1,git add . //2,git commit -mXXX功能 //3,git pull origin new(你们现在这个版本的开发分支) //这里…...
wordpress 定时间隔/赚钱平台
手机APP软件怎么做,就现目前的互联网市场环境来看,主要分为两种,自建技术团队以及选择专业APP外包公司开发,企业商家只有选择合适的开发方式才能开发出更高质量的APP应用程序,下面杭州APP开发公司-七猫网络跟大家详细讲…...
如何在微信上做广告/热狗网站关键词优化
2019独角兽企业重金招聘Python工程师标准>>> 故障描述:论坛访问出现502,开始以为php端口被占用,按常规查端口发现并没有被占用。后又出现mysql无法连接,报(2002) notconnect错误,问百度可能是空间满了。查看…...
网站建设优点/网络营销公司全网推广公司
目录 RequestContext 和上下文处理器 auth debug i18n media static csrf request messages 自动转义 HTML 在单个变量中禁用 在模板中的块里禁用 RequestContext 和上下文处理器 首先,我们快速回顾一下前面介绍的几个术语: • 模板是文本文…...
加强统筹推进政府网站建设/怎么营销自己的产品
深入浅出讲解TCP/UDP协议作者: , 出处:中国电脑教育报, 责任编辑: 许琳, 2005-10-09 16:20图1就是瑞星个人版防火墙软件设置规则的界面。细心的读者会发现,图1中的“协议”栏中有“TCP”、“UDP”等名词,它们是什么意思呢?现在我们就来讲…...
长沙做网站有哪些/百度自然搜索排名优化
2. 概述 MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。MyBatis 可以使用简单的 XML 或注解来配置和映射原生信息, 将接口和 Java 的 POJOs(Plain Old Ja…...
wordpress主题框架Genesis/网上怎么推销自己的产品
在微软平台上,想追求新鲜是越来越不容易了! 昨晚在家,想试一把WPF,到微软网站上去下载orcas最新的版本,发现竟有8个img文件需要下载。想这将来卸载时,也要花不少时间吧,于是就准备下那个base im…...