设计模式-单例设计模式

news/2024/9/15 22:47:33/ 标签: 设计模式, 单例模式

单例模式的设计和线程安全

单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。实现单例模式时,线程安全性是一个重要考虑因素,特别是在多线程环境中。

1. C++11 之前的线程安全实现

在 C++11 之前,确保单例模式的线程安全通常需要使用互斥锁或双重检查锁定(double-checked locking)技术。以下是这些方法的简要说明:

  • 互斥锁:通过使用 std::mutex 或类似的机制来锁定创建实例的代码区域。这样可以确保在多线程环境中,只有一个线程能够执行实例的创建代码,从而避免多个实例的创建。示例如下:

    class Singleton {
    public:static Singleton& getInstance() {if (!instance) {std::lock_guard<std::mutex> guard(mutex); // 加锁if (!instance) {instance = new Singleton();}}return *instance;}private:Singleton() {}static Singleton* instance;static std::mutex mutex;
    };
    

    这个方法避免了多个线程同时创建多个实例的问题,但锁定可能导致性能瓶颈。

  • 双重检查锁定:这种技术用于减少锁的开销。只有在实例尚未创建时才加锁,实例已经创建后则跳过加锁。示例如下:

    class Singleton {
    public:static Singleton& getInstance() {if (!instance) {std::lock_guard<std::mutex> guard(mutex); // 加锁if (!instance) {instance = new Singleton();}}return *instance;}private:Singleton() {}static Singleton* instance;static std::mutex mutex;
    };
    

    这种方法的实现复杂且容易出错,因为在 C++03 及以前版本中,线程对静态变量初始化的行为未被明确规定,可能会导致不一致的结果。

2. C++11 及其之后的改进

C++11 引入了对局部静态变量初始化的线程安全支持,这简化了单例模式的实现。关键改进包括:

  • 线程安全的局部静态变量初始化

    • C++11 标准保证了局部静态变量的初始化是线程安全的。这意味着,即使多个线程同时调用返回静态局部变量的函数,编译器也会确保该静态变量只被初始化一次,并且线程安全。
    • 这消除了对显式锁定的需求,使得单例模式的实现更简单和高效。

    示例代码:

    class Singleton {
    public:static Singleton& getInstance() {static Singleton instance; // 局部静态变量,线程安全return instance;}private:Singleton() {}
    };
    

    在这个实现中,instance 是一个局部静态变量。C++11 确保当多个线程同时访问 getInstance 方法时,只有一个线程会创建 instance 实例,而其他线程将看到已经初始化的实例。

  • 初始化顺序保证

    • C++11 还保证了局部静态变量的初始化顺序是确定的,即先初始化局部静态变量后才执行其他代码。这确保了静态变量不会在使用前被销毁。
3. 总结
  • C++11 改进:引入了线程安全的局部静态变量初始化机制,简化了单例模式的实现,不再需要手动处理线程安全问题。
  • C++11 之前:需要使用互斥锁或双重检查锁定来确保线程安全,这些方法复杂且容易出错。

C++11 的这些改进显著提升了单例模式的实现简洁性和安全性,使得在多线程环境中使用单例模式变得更加可靠和高效。


http://www.ppmy.cn/news/1508719.html

相关文章

电商平台的推荐算法需要备案吗?

答案是肯定的&#xff01; 政策要求&#xff1a; 根据我国《互联网信息服务算法推荐管理规定》&#xff08;以下简称《规定》&#xff09;第六条&#xff0c;具有舆论属性或社会动员能力的互联网信息服务&#xff0c;包括电商平台的推荐算法&#xff0c;需要进行备案。 电商平…

Openlayers6 图形绘制和修改功能(结合React)

Openlayers常用的API了解的差不多了&#xff0c;就开始进入实战了&#xff0c;首先从绘制基本的图形开始&#xff0c;这里主要介绍一下绘制圆形、矩形和多边形。 通过使用openlayers的ol.interaction.Draw和ol.interaction.Modify模块实现地图上绘制圆形、矩形、多边形并修改编…

C++ 函数语义学——inline 函数回顾和扩展

inline 函数回顾和扩展 inline 函数回顾和扩展 inline 函数回顾和扩展1. inline 函数回顾2. inline 扩展总结 1. inline 函数回顾 inline 函数即有优点又有缺点&#xff0c;优点是它的执行成本一般比常规的函数调用和函数返回所带来的成本低&#xff0c;提高了程序执行效率&am…

05_ Electron 自定义菜单、主进程与渲染进程通信

Electron 自定义菜单、主进程与渲染进程通信 一、定义顶部菜单二、Electron 自定义右键菜单1、使用 electron/remote 模块实现 三、 Electron 主进程和渲染进程通信场景1&#xff1a;渲染进程给主进程发送异步消息场景2&#xff1a;渲染进程给主进程发送异步消息&#xff0c;主…

链表 203.移除链表元素 虚拟头结点

普通方法 头节点要和其他节点分开考虑&#xff0c;因为头节点没有上一个节点&#xff0c;不能通过更改上一个结点的指针来达到删除节点的目的 所以要让下一个节点成为头节点&#xff0c;其余节点&#xff0c;通过更改上一个节点的next指针&#xff0c;指向next.next class S…

如何在 Windows 上设置 MacOS 云主机

在Windows上设置MacOS云主机实际上涉及在Windows环境中模拟或远程管理MacOS系统&#xff0c;因为直接在Windows上运行MacOS作为云主机的主操作系统是不可能的&#xff0c;因为MacOS是为苹果硬件设计的。不过&#xff0c;有几种方法可以实现类似的功能&#xff1a; 1. 使用虚拟机…

LVS理论知识

目录 1.描述以及工作原理 1.什么是LVS 2.LVS调度算法 1.静态调度算法 1.轮询RR 2.加权轮询WRR 3.目标地址hash---DH 4.源地址hash---SH 2.动态调度算法 1.LC最少连接 2.wlc加权最少连接 3.sed最少期望延迟 4.nq不排队调度算法 5.lblc基于本地最少连接 6.lnlcr带…

8.8 day bug

bug1 好家伙&#xff0c;最后一个t没看到&#xff0c;愣是学了一个小时原理和用法&#xff0c;都找不出问题在哪

教程:申请IP SSL证书

现在通过IP访问网站的方式也流行起来了&#xff0c;对于很多企业或是组织单位&#xff0c;申请域名不仅需要额外支付域名费用&#xff0c;还需要走备案流程&#xff0c;这对于很多单位来说显得麻烦了一些&#xff0c;所以利用IP直接进行网站访问也是必要的。 但是正常情况下IP…

达梦数据库 逻辑备份还原

达梦的逻辑备份还原 1.背景2.要求3.实验步骤3.1 相关术语3.2 dexp逻辑导出3.2.1 使用dexp工具3.2.2 dexp相关参数含义3.2.3 四种级别导出3.2.3.1 FULL3.2.3.2 OWNER3.2.3.3 SCHEMAS3.2.3.4 TABLES 3.2.4 使用范例3.2.4.1 环境准备3.2.4.2 dexp逻辑导出 3.3 dimp逻辑导入3.3.1 使…

一、软件工程概述

软件工程概述 1. 软件的概念和特点2. 软件危机的产生3. 软件工程的概念和发展过程4. 软件工程知识体系与职业道德 1. 软件的概念和特点 软件定义 软件程序数据文档。 软件生存周期 问题定义&#xff1a;要解决的问题是什么&#xff1f;可行性分析&#xff1a;对于上阶段所确定…

设计模式六大原则(一)——单一职责原则

随着软件开发的复杂度日益增加&#xff0c;维护性和可扩展性成为开发者必须关注的重要因素。设计模式通过提供一系列可重用的解决方案&#xff0c;帮助开发者在设计和实现软件时形成良好的实践。其中&#xff0c;单一职责原则作为设计模式的基石&#xff0c;倡导每个模块、类或…

Tomcat 最大连接数实现原理

spring boot 内置tomcat设置连接数 max-connections: 5 server:port: 9898servlet:context-path: /testtomcat:connection-timeout: 5000max-connections: 5accept-count: 5 ##初始化连接数量connectionLimitLatch protected LimitLatch initializeConnectionLatch() {if (ma…

【K8S】为什么需要Kubernetes?

文章目录 1 什么是Kubernetes&#xff1f;2 三种常见的应用部署方式2.1 传统部署2.2 虚拟化部署2.3 容器化部署 3 Kubernetes的特点写在最后 1 什么是Kubernetes&#xff1f; Kubernetes是 一个开源的&#xff0c;用于管理云平台中多个主机上的容器化应用&#xff0c;Kubernet…

【C++学习】C++中按引用传递与按值传递的具体原理

C中按引用传递与按值传递的具体原理 一、测试代码 void Test(int& a) {a 10; }void TestA(int a) {a 5; }int main() {int a 10;int* b &a;int& c a;a 5;Test(a);TestA(a);*b 3;c 6;std::cout << a << std::endl;std::cin.get(); }二、分析…

图论:欧拉路

欧拉路是什么 什么&#xff1f;你对这个名字感到很陌生&#xff1f;再看看是图论的内容&#xff0c;感觉是不是很难&#xff1f;其实一点也不难&#xff0c;这就是生活中的一笔画问题&#xff0c;也就是不重复的经过每一条边并可以访问所有的点&#xff0c;先看看这个图&#…

【Git】远程仓库新建分支后,拉到本地开发

1. 在远程仓库上创建分支 2. git fetch origin&#xff1a;在本地同步远程仓库的分支&#xff08;获取远程仓库所有分支的所有修改&#xff09; 3. git remote -a&#xff1a;查看所有分支&#xff08;远程&#xff0b;本地&#xff09; 4. git checkout -b 本地名 远程仓库…

SpringMVC学习之 @RequestMapping

关于RequestMapping注解的相关知识&#xff0c;我们这将围绕RequestMapping的功能&#xff0c;位置&#xff0c;method属性以及Params属性、headers属性。 RequestMapping注解的位置 RequestMapping注解的作用就是将请求和处理请求的控制器方法关联起来&#xff0c;建立映射关…

DHCP学习笔记

1.DHCP快速配置 1.1配置接口IP R1: sysname R1 undo info-center enable interface Ethernet0/0/0 ip address 192.168.1.1 255.255.255.0 quit 1.2开启DHCP服务&#xff0c;接着在R1的e0/0/0配置DHCP Server功能 dhcp enable #全局下开启DHCP服务 interface Ethernet…

网络如何发送一个数据包

网络如何发送一个数据包 网络消息发送就是点一点屏幕。 骚瑞&#xff0c;这一点都不好笑。&#xff08;小品就是我的本质惹&#xff09; 之前我就是会被这个问题搞的不安宁。是怎么知道对方的IP地址的呢&#xff1f;怎么知道对方的MAC呢&#xff1f;世界上计算机有那么多&…