C++ 设计模式——单例模式

单例模式

    • C++ 设计模式——单例模式
      • 1. 单例模式的基本概念与实现
      • 2. 多线程环境中的问题
      • 3. 内存管理问题
        • 1. 内存泄漏风险
        • 2. 自动释放策略
        • 3. 垃圾回收机制
        • 4. 嵌套类与内存管理
      • 4. UML 图
        • UML 图解析
      • 优缺点
      • 适用场景
      • 总结

C++ 设计模式——单例模式

单例模式(Singleton Pattern)也称单件模式/单态模式,是一种创建型模式,用于创建只能产生一个对象实例的类。

引入“单例”设计模式的定义(实现意图):保证一个类仅有一个实例存在同时提供能对该实例访问的全局方法(getInstance 成员函数)。

1. 单例模式的基本概念与实现

单例模式通过以下几个关键点实现其目标:

  • 唯一性:利用私有构造函数和静态成员变量,防止外部直接创建类的实例。
  • 全局访问:提供一个公共静态方法(通常命名为 getInstance()),以确保所有调用者都能获取到相同的实例。
  • 懒加载与饿加载:可以选择在类加载时(饿汉式)或首次调用时(懒汉式)创建实例。

实现示例

  • 饿汉式:在类加载时就创建实例,适合对内存占用不敏感的场景。

    class GameConfig {
    private:GameConfig() {};static GameConfig* m_instance;public:static GameConfig* getInstance() {return m_instance;}
    };GameConfig* GameConfig::m_instance = new GameConfig();
    
  • 懒汉式:在首次调用时创建实例,适合资源密集型对象。

    class GameConfig {
    private:GameConfig() {};static GameConfig* m_instance;public:static GameConfig* getInstance() {if (m_instance == nullptr) {m_instance = new GameConfig();}return m_instance;}
    };GameConfig* GameConfig::m_instance = nullptr;
    

2. 多线程环境中的问题

在多线程环境中,懒汉式单例模式可能出现以下问题:

  • 竞态条件:多个线程同时检查实例是否为 nullptr,可能导致多个线程同时创建实例,从而破坏单例特性。
  • 资源浪费:若多个实例被创建,会导致内存和资源的浪费,影响系统性能和稳定性。

解决方案

  • 加锁:在创建实例的代码段中使用互斥锁(如 std::mutex),确保同一时间只有一个线程可以执行实例创建逻辑。
#include <mutex>class GameConfig {
private:GameConfig() {};static GameConfig* m_instance;static std::mutex m_mutex;public:static GameConfig* getInstance() {std::lock_guard<std::mutex> lock(m_mutex); // 加锁if (m_instance == nullptr) {m_instance = new GameConfig();}return m_instance;}
};GameConfig* GameConfig::m_instance = nullptr;
std::mutex GameConfig::m_mutex;
  • 双重检查锁定:在加锁的同时,仍然检查实例是否为 nullptr,以避免不必要的锁开销。
static GameConfig* getInstance() {if (m_instance == nullptr) {std::lock_guard<std::mutex> lock(m_mutex);if (m_instance == nullptr) {m_instance = new GameConfig();}}return m_instance;
}

3. 内存管理问题

单例模式中的内存管理至关重要,尤其是在使用动态分配内存时。以下是一些关键点:

1. 内存泄漏风险
  • 动态分配:如果单例类的实例通过 new 创建,而在程序结束时没有释放内存,可能导致内存泄漏。
  • 手动释放:通常需要提供一个方法(如 freeInstance())来手动释放单例对象的内存。
2. 自动释放策略
  • 使用局部静态变量:在 C++ 中,可以使用局部静态变量来创建单例实例。这种方式的优点是,局部静态变量在程序结束时会自动调用析构函数,释放内存。
class GameConfig {
private:GameConfig() {};GameConfig(const GameConfig&) = delete;GameConfig& operator=(const GameConfig&) = delete;public:static GameConfig& getInstance() {static GameConfig instance; // 自动管理生命周期return instance;}
};
3. 垃圾回收机制
  • 智能指针:使用智能指针(如 std::unique_ptrstd::shared_ptr)来管理单例对象的生命周期,可以减少内存管理的复杂性。
#include <memory>class GameConfig {
private:GameConfig() {};GameConfig(const GameConfig&) = delete;GameConfig& operator=(const GameConfig&) = delete;public:static std::shared_ptr<GameConfig> getInstance() {static std::shared_ptr<GameConfig> instance(new GameConfig());return instance;}
};
4. 嵌套类与内存管理

对于使用饿汉式实现的单例模式,可以引入嵌套类来处理内存释放,确保在程序结束时自动释放内存。

class GameConfig {
private:GameConfig() {};GameConfig(const GameConfig&) = delete;GameConfig& operator=(const GameConfig&) = delete;~GameConfig() {}; // 私有析构函数public:static GameConfig* getInstance() {return m_instance; // 返回静态实例}private:static GameConfig* m_instance; // 指向单例对象的指针// 垃圾回收类class Garbo {public:~Garbo() {if (GameConfig::m_instance != nullptr) {delete GameConfig::m_instance; // 释放内存GameConfig::m_instance = nullptr; // 避免悬空指针}}};static Garbo garboobj; // 静态Garbo对象
};// 静态成员变量初始化
GameConfig* GameConfig::m_instance = new GameConfig(); // 在类外初始化
GameConfig::Garbo GameConfig::garboobj; // 创建Garbo对象

4. UML 图

单例模式 UML 图

UML 图解析
  • 通过私有构造函数和静态成员变量 m_instance,确保 GameConfig 类只有一个实例。
  • 通过公共静态方法 getInstance() 提供全局访问点,允许外部代码获取该实例。
  • 将构造函数和实例变量设为私有,增强了类的封装性,避免了外部对实例的直接操作。

优缺点

优点

  • 唯一性:确保类只有一个实例,避免资源的重复分配。
  • 全局访问:提供全局访问点,使得共享资源的管理更加方便。
  • 延迟实例化:可以实现懒加载,只有在需要时才创建实例,节省资源。

缺点

  • 全局状态:可能导致全局状态的引入,增加系统的耦合性。
  • 难以测试:使得单元测试变得困难,因为单例对象的创建和销毁不够灵活。
  • 多线程问题:在多线程环境下实现复杂,可能引入性能开销和竞态条件。

适用场景

  • 资源共享:适用于需要控制资源的共享,例如配置管理、日志记录和数据库连接等场景。
  • 全局状态管理:适合需要全局访问的状态信息,如应用程序设置、游戏配置等。
  • 限制实例数量:在程序生命周期内只需一个实例的场景,例如线程池、缓存管理和服务注册中心。
  • 懒加载需求:当实例创建较为昂贵且不一定每次都需要时,适合使用懒加载策略。
  • 跨模块访问:需要在多个模块或类中共享同一实例的情况,提升系统的统一性和一致性。

总结

单例模式是一种常用的设计模式,能够有效管理全局资源和状态。通过合理的实现方式,可以避免内存泄漏和多线程问题。理解单例模式的优缺点及适用场景,有助于在实际开发中正确应用这一模式。

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

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

相关文章

【Redis】渐进式遍历和数据库管理

渐进式遍历和数据库管理 渐进式遍历scan 数据库管理切换数据库清除数据库 渐进式遍历 Redis 使⽤ scan 命令进⾏渐进式遍历键&#xff0c;进⽽解决直接使⽤ keys 获取键时可能出现的阻塞问题。每次 scan 命令的时间复杂度是 O(1)&#xff0c;但是要完整地完成所有键的遍历&…

360发布FancyVideo:通过跨帧文本指导实现动态且一致的视频生成SOTA!

文章链接&#xff1a;https://arxiv.org/pdf/2408.08189 项目链接&#xff1a;https://360cvgroup.github.io/FancyVideo/ 亮点直击 本文介绍了FancyVideo&#xff0c;据众所知的首个探索T2V任务中跨帧文本指导的开创性尝试。该方法为增强当前的文本控制方法提供了新的视角。 …

EmguCV学习笔记 VB.Net 6.5 凸包和凸缺陷

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请在显著位置标明本文出处以及作者网名&#xff0c;未经作者允许不得用于商业目的。 EmguCV是一个基于OpenCV的开源免费的跨平台计算机视觉库,它向C#和VB.NET开发者提供了OpenCV库的大部分功能。 教程VB.net版本请访问…

OpenCV c++ 实现图像马赛克效果

VS2022配置OpenCV环境 关于OpenCV在VS2022上配置的教程可以参考&#xff1a;VS2022 配置OpenCV开发环境详细教程 图像马赛克 图像马赛克&#xff08;Image Mosaic&#xff09;的原理基于将图像的特定区域替换为像素块&#xff0c;这些像素块可以是纯色或者平均色&#xff0c…

SpringMVC核心机制环境搭建

文章目录 1.SpringMVC执行流程1.基础流程图2.详细流程图 2.安装Tomcat1.下载2.解压到任意目录即可3.IDEA配置Tomcat1.配置Deloyment2.配置Server 3.创建maven项目1.创建sun-springmvc模块&#xff08;webapp&#xff09;2.查看是否被父模块管理3.pom.xml引入依赖4.目录5.SunDis…

【Redis】Redis数据结构——Hash 哈希

哈希 命令hsethgethexistshdelhkeyshvalshgetallhmgethlenhsetnxhincrbyhincrbyfloat命令小结 内部编码使用场景缓存⽅式对⽐ ⼏乎所有的主流编程语⾔都提供了哈希&#xff08;hash&#xff09;类型&#xff0c;它们的叫法可能是哈希、字典、关联数组、映射。在 Redis 中&#…

C语言函数介绍(上)

函数概念库函数标准库和头文件库函数的使用方法头文件包含库函数文档的一般格式 自定义函数函数的语法形式函数例子 形参和实参实参形参实参和形参的关系 return 语句数组做函数参数 函数概念 数学中我们其实就见过函数的概念&#xff0c;比如&#xff1a;一次函数 ykxb &…

【HuggingFace Transformers】BertModel源码解析

BertModel源码解析 1. BertModel 介绍2. BertModel 源码逐行注释 1. BertModel 介绍 BertModel 是 transformers 库中的核心模型之一&#xff0c;它实现了 BERT&#xff08;Bidirectional Encoder Representations from Transformers&#xff09;模型的架构。BERT 是基于 Trans…

UE5中制作箭头滑动转场

通过程序化的方式&#xff0c;可以制作一些特殊的转场效果&#xff0c;如箭头划过的转场&#xff1a; 1.制作思路 我们知道向量点积可以拿来做投影&#xff0c;因此可以把UV空间想象成向量坐标&#xff0c;绘制结果就是在某个向量上的投影&#xff1a; 绘制结果似乎是倾斜方…

去雾去雨算法

简单版 import cv2 import numpy as npdef dehaze(image):"""简单去雾算法&#xff0c;使用直方图均衡化来增强图像"""# 将图像转换为YUV颜色空间yuv_image cv2.cvtColor(image, cv2.COLOR_BGR2YUV)# 对Y通道&#xff08;亮度&#xff09;进行…

springsecurity 在web中如何获取用户信息(后端/前端)

一、SecurityContextHolder 是什么 SecurityContextHolder用来获取登录之后用户信息。Spring Security 会将登录用户数据保存在Session中。但是&#xff0c;为了使用方便,Spring Security在此基础上还做了一些改进&#xff0c;其中最主要的一个变化就是线程绑定。当用户登录成功…

4820道西医综合真题西医真题ACCESS\EXCEL数据库

本题库内容源自某出版物《西医综合真题考点还原与答案解析》&#xff0c;包含4千多道真题。这个数据库包含3个表&#xff0c;一个是分类表&#xff08;SECTION_BEAN&#xff09;&#xff0c;一个是题库主表&#xff08;QUESTION_INFO_BEAN&#xff09;&#xff0c;一个是选项表…

【网络】HTTP

在上一篇文章中&#xff0c;我们了解了 协议 的制定与使用流程&#xff0c;不过太过于简陋了&#xff0c;真正的 协议 会复杂得多&#xff0c;也强大得多&#xff0c;比如在网络中使用最为广泛的 HTTP/HTTPS 超文本传输协议 但凡是使用浏览器进行互联网冲浪&#xff0c;那必然…

【生物特征识别论文分享】基于深度学习的掌纹掌静脉识别

&#xff08;待更新&#xff09;基于深度学习的生物特征识别&#xff08;手掌静脉、手背静脉、手指静脉、掌纹、人脸等&#xff09;论文模型总结 。具体方法包括&#xff1a;基于特征表征、基于传统网络设计与优化、基于轻量级网络设计与优化、基于Transformer设计与优化、基于…

Leetcode 100.101.110.199 二叉树相同/对称/平衡 C++实现

Leetcode 100. 相同的树 问题&#xff1a;给你两棵二叉树的根节点 p 和 q &#xff0c;编写一个函数来检验这两棵树是否相同。 如果两个树在结构上相同&#xff0c;并且节点具有相同的值&#xff0c;则认为它们是相同的。 /*** Definition for a binary tree node.* struct T…

Error: Can not import paddle core while this file exists

背景 因为工作需要&#xff0c;原来的项目部署的电脑被征用&#xff0c;重新换了一个新电脑&#xff0c;重装了系统&#xff0c;今天在给一个使用ocr的项目进行环境配置的时候发现&#xff0c;无论安装哪个版本的paddlepaddle&#xff0c;总是可以安装成功&#xff0c;但是导入…

开源接口自动化测试工具AutoMeter

AutoMeter是一款针对分布式服务和微服务API做功能和性能一体化的自动化测试平台。一站式提供项目管理&#xff0c;微服务&#xff0c;API接口&#xff0c;用例&#xff0c;环境管理&#xff0c;测试管理&#xff0c;前置条件&#xff0c;测试集合&#xff0c;变量管理&#xff…

kali安装

引言 Kali Linux 是一个基于 Debian 的 Linux 发行版&#xff0c;专门为渗透测试和安全审计而设计。它包含了大量的安全工具&#xff0c;如 Wireshark、Nmap、Metasploit 等&#xff0c;这些工具可以帮助安全专家和研究人员进行网络安全评估、漏洞检测和渗透测试。Kali Linux …

系统架构师(每日一练23)

每日一练 1.软件活动主要包括软件描述、()、软件有效性验证和()&#xff0c;()定义了软件功能及使用限制。答案与解析 问题1 A.软件模型 B.软件需求 C.软件分析 D.软件开发 问题2 A.软件分析 B.软件测试 C.软件演化 D.软件开发 问题3 A.软件分析 B.软件测试 C.软件描述 D.软…