C++设计模式_12_Singleton 单件模式
在之前的博文C++57个入门知识点_44:单例的实现与理解中,已经详细介绍了单例模式,并且根据其中内容,单例模式已经可以在日常编码中被使用,本文将会再做梳理。
Singleton 单件模式可以说是最简单的设计模式,但由于多线程环境的双检查锁里的内存reorder的问题,实现时的细节并不简单,大家需要注意多线程环境下的安全做法。为了整体的一致性,本篇都称之为单件模式。
文章目录
- 1. “对象性能”模式
- 1.1 典型模式
- 2. 动机(Motivation)
- 3. Singleton 单件模式的实现版本
- 3.1 版本1:线程非安全版本
- 3.2 版本2:线程安全版本,但锁的代价过高
- 3.3 版本3:双检查锁
- 3.4 版本4:C++ 11版本之后的跨平台实现
- 3.5 总结
- 4. 模式定义
- 5. 结构
- 6. 要点总结
- 7. 其他参考
单件模式属于一个新的类别,将其归结到“对象性能”
模式,该模式的简介如下:
1. “对象性能”模式
面向对象很好地解决了“抽象”
的问题,但是不可避免地要付出一定的代价(虚函数(内存等代价哒)、继承)。对于通常情况来讲,面向对象的成本大都可以忽略不计。但是某些情况(倍乘效应等),面向对象所带来的成本必须谨慎处理。
1.1 典型模式
- Singleton
- Flyweight
2. 动机(Motivation)
- 在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性、以及良好的效率
- 如何绕过常规的构造器,提供一种机制来保证一个类只有一个实例?
- 这应该是类设计者的责任,而不是使用者的责任
3. Singleton 单件模式的实现版本
首先将构造函数和拷贝构造函数设置为私有,如果不写的话,编译器会默认生成两个公有的,让外界不能使用它们,唯一的做法就是将其设置为私有的。然后再设置静态变量,并进行静态变量的初始化。代码如下
class Singleton{
private:Singleton();Singleton(const Singleton& other);
public:static Singleton* getInstance();static Singleton* m_instance;
};
3.1 版本1:线程非安全版本
先来看第一种写法,第一次调用的时候,m_instance == nullptr
成立的情况下m_instance = new Singleton();
创建对象,返回m_instance,第二次调用的话,m_instance == nullptr
不成立,继续返回原先的对象,这样就可以保证永远只返回第一次的对象。
//线程非安全版本
Singleton* Singleton::getInstance() {if (m_instance == nullptr) {m_instance = new Singleton();}return m_instance;
}
上述的版本会有其他问题,它在单线程下是OK的,但是多线程环境下就不行了。假设我有2个thread,threadA进到if (m_instance == nullptr)
判断第一次调用,还没执行下一行,threadB假如此时得到时间片,开始执行if (m_instance == nullptr)
也是成立,就会执行下一行,此时再跳回threadA,此时就会继续执行下一行,这样就会产生2个对象,因此这是线程非安全的。为了实现多线程安全如何去做呢?
3.2 版本2:线程安全版本,但锁的代价过高
第一种常见做法:加锁
//线程安全版本,但锁的代价过高
Singleton* Singleton::getInstance() {Lock lock; //锁的局部变量,出作用域之后消失,释放锁if (m_instance == nullptr) {m_instance = new Singleton();}return m_instance;
}
TreadA进到Lock lock;
开始加锁,往下执行时,另外一个线程threadB进来,但是TreadA已经获取到锁,仍未释放,threadB就需要一直等到TreadA执行完函数释放锁。等threadB进去之后m_instance == nullptr
就不成立了。
3.3 版本3:双检查锁
上一种版本实现了锁的安全,但是锁的代价太高。假如对象已经创建,那么有两个线程同时访问,threadA执行到Lock lock;
,threadB是无法进入的,threadB是一个读操作,当存在多个线程的情况下,读操作是不需要加锁的
。读操作是没有安全问题的,这个时候加锁对于读操作的线程是一种等待时间上的浪费。假如高并发场景下,web服务是典型高并发,10万人同时访问时,锁对于读操作代价也会是十分高的。
所以发展出一种新的实现形式,double check lock 双检查锁,之前的版本是不管什么情况,先锁,现在是if(m_instance==nullptr)
先问下变量是否为空值,如果为空才加锁,如果不空就不需要加锁,直接返回即可,加完锁之后再判断是否为空。这样就在读操作时减少了时间代价。
//双检查锁,但由于内存读写reorder不安全
Singleton* Singleton::getInstance() {if(m_instance==nullptr){Lock lock;if (m_instance == nullptr) {m_instance = new Singleton();}}return m_instance;
}
有人会考虑,第二个判断if (m_instance == nullptr)
是不是可以去掉,我们假设2个线程,threadA经过第一个判断进来,然后在执行Lock lock;
之前,threadB也会进来到Lock lock;
之前,两个线程即使加锁也就会等一等,都会执行代码m_instance = new Singleton();
,这样就会有2个对象。第二个判断就是为了放置这种情况的出现。
上面的版本看起来就是没有问题的了,也相当一段时间内迷惑了计算机领域的专家,大概2000年时才被JAVA领域的专家发现漏洞:内存读写reorder
导致双检查锁的失效。
什么是reorder,正常情况下,大家看到的代码会有指令序列,那么大家通常会认为,指令序列会按照你所想象的方式执行,但是实际上整个代码到了CPU的指令层次(线程是在指令层次抢时间片),指令和我们很多时候的假设很可能不一样。比如: m_instance = new Singleton();
一般会有以下的假设,拆分成3个步骤,第一步是先分配内存,第二步是调用构造器(对分配的内存进行初始化),第三步把内存的地址(指针)给m_instance。但是这三步是我们假想的,但是在指令级别,很可能被编译器优化reorder,在reorder之后可能就会先第一步,第三步,最后再第三步。很多研究者在不同的CPU上做实验发现很多语言都会这种现象,这样一来就有些乱套。按照可能的reorder顺序,先分配内存,直接就赋值,m_instance就不是nullptr了,构造器还没调用,另一个线程做第一次判断时发现m_instance不是nullptr,直接返回一个对象,返回的这个对象还没有经过构造器处理,其状态不对,是不可用的。
3.4 版本4:C++ 11版本之后的跨平台实现
为了解决上面的问题,java和c#语言就加了volatile的关键字来保证按照我们的想法来执行。VC++中自己也加了volatile的关键字。
C++11之后才有了这样的机制,可以帮助可以在语言层面实现跨平台的实现。
//C++ 11版本之后的跨平台实现 (volatile)
std::atomic<Singleton*> Singleton::m_instance; //声明原子对象
std::mutex Singleton::m_mutex;Singleton* Singleton::getInstance() {Singleton* tmp = m_instance.load(std::memory_order_relaxed);//得到tmp指针,帮助实现屏蔽编译器的reorderstd::atomic_thread_fence(std::memory_order_acquire);//获取内存fence,内存reorder的屏障//tmp就不会被reorder,底下为双检查锁的操作if (tmp == nullptr) {std::lock_guard<std::mutex> lock(m_mutex);tmp = m_instance.load(std::memory_order_relaxed);if (tmp == nullptr) {tmp = new Singleton;std::atomic_thread_fence(std::memory_order_release);//释放内存fencem_instance.store(tmp, std::memory_order_relaxed);}}return tmp;
}
经过上面的操作,才在C++11之后实现跨平台的实现。
3.5 总结
总体来说,如果是单线程使用版本1已经是足够好了;如果是多线程,使用版本2是OK的,但是效率不高;双检查锁在不使用volatile不能用,是很容易出现问题的;C++11版本之后跨平台实现双检查锁就使用的是最后一个版本
4. 模式定义
保证一个类仅有一个实例,并提供一个该实例的全局访问点
–《设计模式》GoF
5. 结构
6. 要点总结
- Singleton模式中的实例构造器可以设置为protected以允许子类派生
- Singleton模式一般不要支持拷贝构造函数和Clone接口,因为这有可能导致多个对象实例,与Singleton模式的初衷违背。
- 如何实现多线程环境下安全的Singleton?注意对双检查锁的正确实现。
7. 其他参考
C++设计模式——单例模式
相关文章:

C++设计模式_12_Singleton 单件模式
在之前的博文C57个入门知识点_44:单例的实现与理解中,已经详细介绍了单例模式,并且根据其中内容,单例模式已经可以在日常编码中被使用,本文将会再做梳理。 Singleton 单件模式可以说是最简单的设计模式,但由…...

67 内网安全-域横向smbwmi明文或hash传递
#知识点1: windows2012以上版本默认关闭wdigest,攻击者无法从内存中获取明文密码windows2012以下版本如安装KB2871997补丁,同样也会导致无法获取明文密码针对以上情况,我们提供了4种方式解决此类问题 1.利用哈希hash传递(pth,ptk等…...

面向对象(类/继承/封装/多态)详解
简介: 面向对象编程(Object-Oriented Programming,OOP)是一种广泛应用于软件开发的编程范式。它基于一系列核心概念,包括类、继承、封装和多态。在这篇详细的解释中,我们将探讨这些概念,并说明它们如何在P…...

【Python机器学习】零基础掌握GradientBoostingRegressor集成学习
如何精准预测房价? 当人们提到房价预测时,很多人可能会想到房地产经纪人或专业的评估师。但是,有没有一种更科学、更精确的方法来预测房价呢?答案是有的,这就要用到机器学习中的一种算法——梯度提升回归(Gradient Boosting Regressor)。 假设现在有一组房屋数据,包括…...

【tio-websocket】12、应用层包—Packet
Packet 介绍 Packet 是用于表述业务数据结构的,我们通过继承 Packet 来实现自己的业务数据结构,对于各位而言,把 Packet 看作是一个普通的 VO 对象即可。 public class Packet implements java.io.Serializable, Cloneable {private static Logger log = LoggerFac…...

OpenCV官方教程中文版 —— 模板匹配
OpenCV官方教程中文版 —— 模板匹配 前言一、原理二、OpenCV 中的模板匹配三、多对象的模板匹配 前言 在本节我们要学习: 使用模板匹配在一幅图像中查找目标 函数:cv2.matchTemplate(),cv2.minMaxLoc() 一、原理 模板匹配是用来在一副大…...

如何为3D模型设置自发光材质?
1、自发光贴图的原理 自发光贴图是一种纹理贴图,用于模拟物体自发光的效果。其原理基于光的发射和反射过程。 在真实世界中,物体自发光通常是由于其本身具有能够产生光的属性,如荧光物质、发光材料或光源本身。为了在计算机图形中模拟这种效…...

UI组件库基础
UI组件库 全局组件* 全局注册组件 & 并且使用了require.context 模块化编程 & webpack打包 const install(Vue)>{const contextrequire.context(.,true,/\.vue$/)context.keys().forEach(fileName>{const modulecontext(fileName)Vue.component(module.default.n…...

数据结构与算法之矩阵: Leetcode 48. 旋转矩阵 (Typescript版)
旋转图像 https://leetcode.cn/problems/rotate-image/ 描述 给定一个 n n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。你必须在 原地 旋转图像,这意味着你需要直接修改输入的二维矩阵。请不要 使用另一个矩阵来旋转图像。 示例 1 输入&…...

大厂面试题-JVM中的三色标记法是什么?
目录 问题分析 问题答案 问题分析 三色标记法是Java虚拟机(JVM)中垃圾回收算法的一种,主要用来标记内存中存活和需要回收的对象。 它的好处是,可以让JVM不发生或仅短时间发生STW(Stop The World),从而达到清除JVM内存垃圾的目的ÿ…...

Leetcode—121.买卖股票的最佳时机【简单】
2023每日刷题(十一) Leetcode—17.电话号码的字母组合 枚举法题解 参考自灵茶山艾府 枚举法实现代码 int maxProfit(int* prices, int pricesSize){int i;int max 0;int minPrice prices[0];for(i 1; i < pricesSize; i) {int tmp prices[i] -…...

【云原生】portainer管理多个独立docker服务器
目录 一、portainer简介 二、安装Portainer 1.1 内网环境下: 1.1.1 方式1:命令行运行 1.1.2 方式2:通过compose-file来启动 2.1 配置本地主机(node-1) 3.1 配置其他主机(被node-1管理的节点服务器&…...

Command集合
Command集合 mysql相关命令 查看mysql的状态 sudo netstat -tap | grep mysql 启动mysql sudo service mysql start 停止mysql sudo service mysql stop 重启mysql sudo service mysql restart 指定端口号,客户端连接mysql sudo mysql -h127.0.0.1 -uroot -p red…...

【QT开发(17)】2023-QT 5.14.2实现Android开发
1、简介 搭建Qt For Android开发环境需要安装的软件有: JAVA SDK (jdk 有apt install 安装) Android SDK Android NDKQT官网的介绍: Different Qt versions depend on different NDK versions, as listed below: Qt versionNDK…...

JVM相关面试题(每日一练)
1. 什么是垃圾回收机制? 垃圾收集 Garbage Collection 通常被称为“GC”,它诞生于1960年 MIT 的 Lisp 语言,经过半个多世纪,目前已经十分成熟了。 jvm 中,程序计数器、虚拟机栈、本地方法栈都是随线程而生随线程而灭&a…...

OpenCV 相机相关函数
一、变换参数矩阵的求解 1. 计算三个二维点对之间的仿射变换矩阵:getAffineTransform() 2. 计算多个二维点对之间的最优放射变换矩阵(误差最小准则):estimateRigidTransform();或者findHomography(); 3. 计算四个二维点对之间的…...

微信小程序之投票管理
前言 对于会议管理模块,必不可少的当然就是我们的投票管理,实现真正意义上的无纸化办公,本期博客为大家介绍会议管理模块,包括发布投票及查看各类投票的状态 所用技术点 MyBatis、SpringMVC、VentUI MyBatis和SpringMVC在博客主…...

23种设计模式【创建型模式】详细介绍之【建造者模式】
建造者模式:构建复杂对象的精妙设计 设计模式的分类和应用场景总结建造者模式:构建复杂对象的精妙设计建造者模式的核心思想建造者模式的参与者Java示例:建造者模式 设计模式的分类和应用场景总结 可以查看专栏设计模式:设计模式 …...

[量化投资-学习笔记002]Python+TDengine从零开始搭建量化分析平台-MA均线的多种实现方式
MA 均线时最基本的技术指标,也是最简单,最不常用的(通常使用EMA、SMA)。 以下用两种不同的计算方法和两种不同的画图方法进行展示和说明。 MA 均线指标公式 MA (N)(C1 C2 C3 …C N )/N目录 方式一1.SQL 直接查询均值2.使用 pyp…...

c语言 判断两个文件是否相同
使用strcmp比较: #include <stdio.h> #include <string.h>int Compare(const char * file1, const char* file2) {FILE* f1, * f2;int size1, size2;unsigned char buffer1[1024], buffer2[1024];f1 fopen(file1, "rb");f2 fopen(file2, &…...

【2021集创赛】Arm杯三等奖:基于FPGA的人脸检测SoC设计
本作品参与极术社区组织的有奖征集|秀出你的集创赛作品风采,免费电子产品等你拿~活动。 团队介绍 参赛单位:合肥工业大学 队伍名称:芯创之家 指导老师:邓红辉、尹勇生 参赛杯赛:Arm杯 参赛人员:王亮 李嘉燊 金京 获奖情…...

Java电商平台 - API 接口设计之 token、timestamp、sign 具体架构与实现|电商API接口接入
一:token 简介 Token:访问令牌access token, 用于接口中, 用于标识接口调用者的身份、凭证,减少用户名和密码的传输次数。一般情况下客户端(接口调用方)需要先向服务器端申请一个接口调用的账号,服务器会给出一个appId和一个key, …...

【带头学C++】----- 1.基础知识 ---- 1.23 运算符概述
1.23 运算符概述 运算符,在数学中常见的加减乘除之类的符号,那么在C在编程语言中呢,将使用特定的符号或标记对操作数进行操作以生成结果。用算术运算符将运算对象(也称操作数)连接起来的、符合C 语法规则的式子,称为C 算术表达式运…...

python爬虫分析基于python图书馆书目推荐数据分析与可视化
收藏关注不迷路 文章目录 前言一、项目介绍二、开发环境三、功能介绍四、核心代码五、效果图六、文章目录 前言 随着电子技术的普及和快速发展,线上管理系统被广泛的使用,有很多商业机构都在实现电子信息化管理,图书推荐也不例外,…...

Java零基础入门-关系运算符
前言 Java作为一门广受欢迎的开发语言,其在企业级应用和移动应用开发中有着广泛的应用。如果你是一个Java零基础的初学者,那么你来到了一个正确的地方。在本篇文章中,我们会详细介绍Java中的关系运算符,帮助你快速入门。 摘要 …...

1200*A. Trust Nobody(贪心)
Problem - 1826A - Codeforces 解析: 从大到小枚举说谎人的个数x,然后查看是否有 x个人说谎即可。 #include<bits/stdc.h> using namespace std; #define int long long const int N2e55; int t,n,a[N]; signed main(){scanf("%lld",&a…...

二维码智慧门牌管理系统升级解决方案:采集项目的建立与运用
文章目录 前言一、采集项目的建立二、采集项目的运用三、采集项目的意义 前言 在二维码智慧门牌管理系统的升级过程中,一个至关重要的环节是采集项目的建立与运用。采集项目是新建采集任务的前提,同时也是整个系统升级的关键步骤。其意义近似于现实中的…...

Azure - 机器学习:创建机器学习所需资源,配置工作区
目录 一、Azure机器学习工作区与计算实例简要介绍工作区计算实例 二、创建工作区1. 登录到 Azure 机器学习工作室2. 选择“创建工作区”3. 提供以下信息来配置新工作区:4. 选择“创建”以创建工作区 三、创建计算实例四、工作室实战4.1 工作室快速导览4.2 从示例笔记…...

电脑监控软件哪些比较好用
电脑监控软件在当今信息化时代越来越受到人们的关注,它们可以用于保护公司的商业机密,防止员工在工作中做一些不恰当的事情,以及在家庭中监控孩子的上网行为等。 本文将介绍一些比较好用的电脑监控软件: 一、域之盾软件 这款软件…...

数据结构与算法之排序: 选择排序 (Javascript版)
排序 排序:把某个乱序的数组变成升序或降序的数组 (这里用数组来做举例) 选择排序 该排序属于 贪心 策略关注的是局部,是一种苟且的东西 算法实现 // 随机数组,选择排序 Array.prototype.selectionSort function() {let len this.leng…...