【C++入门到精通】特殊类的设计 | 单例模式 [ C++入门 ]

在这里插入图片描述

阅读导航

  • 引言
  • 一、设计模式概念(了解)
  • 二、单例模式
    • 1. 饿汉模式
      • (1)概念
      • (2)模拟实现
      • (3)优缺点
      • (4)适用场景
    • 2. 懒汉模式
      • (1)概念
      • (2)模拟实现
        • 🚩思路一(双检查加锁,常规思路)
        • 🚩思路二(使用静态局部变量的方式来实现单例模式)
      • (3)优缺点
      • (4)适用场景
  • 温馨提示

引言

在面向对象编程中,特殊类是指具有特定属性或限制的类,这些属性或限制使其在设计和使用上与常规类不同。在上一篇文章中,我们讨论了一些特殊类,如只能在堆上创建对象的类、只能在栈上创建对象的类以及禁止拷贝和继承的类。

在本文中,我们将继续探讨特殊类的设计,着重介绍单例模式。单例模式是一种常见的设计模式,它确保一个类只有一个实例,并提供了全局访问点。在许多情况下,我们需要确保只有一个对象来协调系统操作或管理共享资源,而单例模式正是解决这类问题的理想选择

本文将深入研究单例模式的原理和实现方式。我们将介绍几种常见的单例模式实现方法,包括饿汉式、懒汉式、双重检查锁定和静态内部类。我们将详细讨论每种实现方法的优缺点,并提供相应的示例代码。让我们一起探索单例模式的精髓吧!

一、设计模式概念(了解)

设计模式是一种被广泛接受和应用的软件开发经验总结,它提供了解决常见问题的可重用方案。设计模式帮助开发人员以一种可靠、灵活和可维护的方式构建软件系统。

设计模式的概念最早由计算机科学家埃里希·伽玛Erich Gamma)等人在1994年的著作《设计模式:可复用面向对象软件的基础》中引入。该书提出了23种经典的设计模式,这些模式分为三大类:创建型模式、结构型模式和行为型模式。

每种设计模式都有其特定的应用场景和解决方案,开发人员可以根据具体需求选择适当的模式来解决问题。设计模式不仅提供了一种通用的解决方案,还促进了代码的可读性、可维护性和可扩展性。

然而,设计模式并非万能药,过度使用或错误使用设计模式可能导致代码变得复杂和难以理解。因此,在应用设计模式时,开发人员需要谨慎权衡,并结合实际情况做出决策。

总之,设计模式是一种帮助开发人员解决常见问题的工具,它提供了一套经过验证的解决方案。通过学习和应用设计模式,开发人员可以提高软件系统的质量和可维护性,从而更加高效地开发出优秀的软件。使用设计模式的目的:为了代码可重用性、让代码更容易被他人理解、保证代码可靠性。 设计模式使代码编写真正工程化;设计模式是软件工程的基石脉络,如同大厦的结构一样

二、单例模式

一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。

⭕单例模式有两种实现模式:饿汉模式懒汉模式,下面我会一个一个的向大家介绍

1. 饿汉模式

(1)概念

饿汉模式是单例模式的一种实现方式,它在类加载时就创建唯一的实例对象,并通过静态方法提供全局访问点。简单来说就是不管你将来用不用,程序启动时就创建一个唯一的实例对象

🍪特点

  • 在类加载时就创建实例对象,因此可以保证实例的唯一性。
  • 通过静态方法提供全局访问点,方便其他代码获取该实例。
  • 线程安全,由于在类加载时创建实例,因此不需要考虑多线程并发访问的问题。

(2)模拟实现

// 饿汉模式:一开始(main函数之前)就创建对象
class Singleton
{
public:// 静态方法,返回唯一实例对象的地址static Singleton* GetInstance(){return _ins;}// 向字符串向量中添加元素,保证线程安全void Add(const string& str){_mtx.lock();    // 获取互斥锁_v.push_back(str);  // 执行操作_mtx.unlock();  // 释放互斥锁}// 打印字符串向量中的所有元素,保证线程安全void Print(){_mtx.lock();    // 获取互斥锁for (auto& e : _v){cout << e << endl;}cout << endl;_mtx.unlock();  // 释放互斥锁}private:// 构造函数私有化,禁止外部创建对象Singleton(){}// 防拷贝构造和赋值运算符,保证实例的唯一性Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;private:mutex _mtx;         // 互斥锁,保证线程安全vector<string> _v;  // 字符串向量,存储数据static Singleton* _ins; // 唯一实例对象的地址
};// 初始化静态成员变量
Singleton* Singleton::_ins = new Singleton();

以上代码实现了一个简单的使用饿汉模式实现的线程安全的单例类。它在类加载时就创建了唯一的实例对象,并提供了全局访问点,适用于需要在整个应用程序中共享一个实例对象的场景。同时,该类的实现还保证了多线程并发访问时的线程安全性,避免了数据竞争和死锁等问题。

(3)优缺点

  • 优点

    1. 实现简单直观,代码易于理解
    2. 线程安全,不需要额外的同步处理,适合在多线程环境中使用。
    3. 对象的创建是在类加载时完成的,可以避免线程安全问题和延迟加载的复杂性
  • 缺点

    1. 在程序运行期间始终存在实例对象,可能会造成资源浪费
    2. 如果该实例对象的创建过程耗时较长,会导致应用程序启动变慢
    3. 不支持延迟加载,无法根据实际需要来创建实例

(4)适用场景

  • 对象的创建过程简单且耗时较短的情况,适合使用饿汉模式。
  • 需要在整个应用程序中共享一个实例对象的情况,适合使用饿汉模式。
  • 在多线程环境下需要保证实例的唯一性和线程安全的情况,适合使用饿汉模式。

总的来说:饿汉模式是一种简单有效的单例模式实现方式,适合于对象创建耗时较短、且需要全局访问的情况。但在实际应用中,需要根据具体需求和性能要求选择适当的单例模式实现方式

2. 懒汉模式

(1)概念

懒汉模式是指在需要时才创建实例对象的单例模式在懒汉模式中,实例对象的创建被延迟到第一次使用时,而不是在程序启动时就立即创建。这样可以避免在程序启动时创建不必要的实例对象,节省系统资源

如果单例对象的构造过程耗时且资源占用较多,例如加载插件、初始化网络连接或读取文件等操作,同时在程序运行过程中可能并不经常使用该对象,那么在程序启动时立即进行初始化会导致启动速度缓慢。因此,在这种情况下,采用懒汉模式(延迟加载)是更好的选择

⭕懒汉模式允许在需要使用该对象时才创建实例,避免了不必要的资源浪费,提高了程序性能。通过懒汉模式,可以延迟加载单例对象,无需在程序启动时进行初始化,从而避免了启动时的缓慢问题。

(2)模拟实现

🚩思路一(双检查加锁,常规思路)
class Singleton
{
public:static Singleton* GetInstance(){// 双检查加锁,提高效率if (_ins == nullptr)  // 第一次检查{_imtx.lock();  // 加锁if (_ins == nullptr)  // 第二次检查,确保线程安全{_ins = new Singleton;  // 创建单例对象}_imtx.unlock();  // 解锁}return _ins;}// 显示释放单例对象static void DelInstance(){_imtx.lock();  // 加锁if (_ins){delete _ins;  // 释放单例对象_ins = nullptr;  // 将指针置为空}_imtx.unlock();  // 解锁}// 内部类:用于单例对象的资源回收和持久化class GC{public:~GC(){DelInstance();  // 调用DelInstance()函数进行资源回收和持久化}};// 内部静态成员变量,用于实现单例模式static Singleton* _ins;static mutex _imtx;  // 互斥锁,保证线程安全// 添加数据到vector中void Add(const string& str){_vmtx.lock();  // 加锁_v.push_back(str);  // 添加数据到vector中_vmtx.unlock();  // 解锁}// 输出vector中的数据void Print(){_vmtx.lock();  // 加锁for (auto& e : _v){cout << e << endl;  // 输出vector中的数据}cout << endl;_vmtx.unlock();  // 解锁}// 析构函数,用于实现单例对象的持久化~Singleton(){// 比如要求程序结束时,将数据写到文件,单例对象析构时持久化就比较好}private:// 私有构造函数,限制类外部创建对象Singleton(){}// 防拷贝Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;mutex _vmtx;  // 互斥锁,保证线程安全vector<string> _v;  // 存储数据的vectorstatic GC _gc;  // 内部类对象,用于单例对象析构时进行资源回收和持久化
};// 初始化静态成员变量
Singleton* Singleton::_ins = nullptr;
mutex Singleton::_imtx;
Singleton::GC Singleton::_gc;
🚩思路二(使用静态局部变量的方式来实现单例模式)
class Singleton
{
public:// 获取单例对象的接口函数static Singleton* GetInstance(){// 使用静态局部变量实现单例模式,保证线程安全// C++11之前,这里不能保证初始化静态对象的线程安全问题// C++11之后,这里可以保证初始化静态对象的线程安全问题static Singleton inst;return &inst;}// 添加数据到vector中void Add(const string& str){// 加锁,保证线程安全_vmtx.lock();// 添加数据到vector中_v.push_back(str);// 解锁,保证线程安全_vmtx.unlock();}// 输出vector中的数据void Print(){// 加锁,保证线程安全_vmtx.lock();// 遍历vector,输出其中的元素for (auto& e : _v){cout << e << endl;}cout << endl;// 解锁,保证线程安全_vmtx.unlock();}// 析构函数,用于实现单例对象的持久化~Singleton(){// 比如要求程序结束时,将数据写到文件,单例对象析构时持久化就比较好}private:// 私有构造函数,限制类外部创建对象Singleton(){cout << "Singleton()" << endl;}// 防拷贝Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;private:mutex _vmtx;  // 互斥锁,保证线程安全vector<string> _v;  // 存储数据的vector
};

这段代码是使用静态局部变量的方式来实现单例模式。

🚨🚨注意:在 C++11 之前,使用静态局部变量的方式需要注意线程安全问题,因为静态局部变量的初始化只会在第一次调用时进行,如果有多个线程同时调用,可能会导致不同步的问题。但在 C++11 之后,静态局部变量的初始化是线程安全的,因此可以放心使用

(3)优缺点

  • 优点

    1. 延迟加载:懒汉模式在需要时才创建实例对象,避免了在程序启动时的资源浪费。这对于资源消耗较大的对象特别有用。
    2. 节省系统资源:由于实例对象的创建被延迟到需要时,懒汉模式可以节省系统资源,提高程序的性能。
    3. 线程安全:通过双重判断锁机制,懒汉模式可以在多线程环境下保证线程安全性。
  • 缺点

    1. 复杂性增加:相比饿汉模式,懒汉模式的实现相对复杂,需要考虑线程安全性问题。
    2. 性能损耗:在多线程环境下,由于需要进行双重判断锁机制,可能会导致一定的性能损耗。

(4)适用场景

  • 对象创建耗时较长或占用较多资源:懒汉模式可以避免在程序启动时创建不必要的实例对象,节省系统资源。
  • 需要延迟加载的场景:如果单例对象在程序运行的早期并不会被频繁使用,而只有在特定条件下才会被需要,那么懒汉模式是一个合适的选择。
  • 多线程环境下需要保证线程安全性:通过双重判断锁机制,懒汉模式可以在多线程环境下保证线程安全性,避免多个线程同时创建多个实例对象。

温馨提示

感谢您对博主文章的关注与支持!另外,我计划在未来的更新中持续探讨与本文相关的内容,会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!

再次感谢您的支持和关注。期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

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

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

相关文章

SpringBoot 结合 liteflow 规则引擎使用

1、前言 在日常的开发过程中&#xff0c;经常会遇到一些串行或者并行的业务流程问题&#xff0c;而业务之间不必存在相关性。 在这样的场景下&#xff0c;使用策略和模板模式的结合可以很好的解决这个问题&#xff0c;但是使用编码的方式会使得文件太多,在业务的部分环节可以…

利用操作符解题的精彩瞬间

下面是链接为了解释练习2的并且还有与操作符相关的知识。 C语言与操作符相关的经典例题-CSDN博客 操作符详解&#xff08;上&#xff09;-CSDN博客 操作符详解&#xff08;下&#xff09;-CSDN博客 目录 练习1&#xff1a;在一个整型数组中&#xff0c;只有一个数字出现一…

burp靶场--xss下篇【16-30】

burp靶场–xss下篇【16-30】 https://portswigger.net/web-security/all-labs#cross-site-scripting 实验16&#xff1a;允许使用一些 SVG 标记的反射型 XSS ### 实验要求&#xff1a; 该实验室有一个简单的反射型 XSS漏洞。该网站阻止了常见标签&#xff0c;但错过了一些 S…

【Midjourney】关于标准模型的几个按钮都有什么用

当用户在Midjourney Bot所在的服务发送/settings命令时就能调出设置窗口&#xff0c;本文将介绍该窗口中的各个按钮都有什么作用。 1.RAW Mode 依照官方的描述来看V5.2模型似乎带有自动优化功能&#xff0c;会对用户输入的关键词空白描述进行补全和优化&#xff0c;以便修复所…

如何快速记忆小鹤双拼键位图?

记忆方法&#xff1a;韵母表 图形 最常用字 韵母表&#xff1a;双拼的基础 图形&#xff1a;帮助新手快速联想回忆 最常用字&#xff1a;快速打字基础 一、单韵母&#xff08;紫色方块&#xff09; 一一对应如下表&#xff1a; 单韵母aoeiu、AOEIV 二、复韵母—箭矢型&am…

【springboot网页时装购物系统】

前言 &#x1f31e;博主介绍&#xff1a;✌全网粉丝15W,CSDN特邀作者、211毕业、高级全栈开发程序员、大厂多年工作经验、码云/掘金/华为云/阿里云/InfoQ/StackOverflow/github等平台优质作者、专注于Java、小程序技术领域和毕业项目实战&#xff0c;以及程序定制化开发、全栈…

人工智能与机器学习——开启智能时代的里程碑

写在前面 前言人工智能与机器学习的概述监督学习、无监督学习和强化学习的基本原理监督学习&#xff1a;无监督学习&#xff1a;强化学习&#xff1a; 机器学习的算法和方法常见的机器学习算法和方法线性回归&#xff1a;决策树&#xff1a;支持向量机&#xff1a;神经网络&…

Ubuntu20.04添加桌面启动、侧边栏启动和终端启动

桌面启动 新建XX.desktop文件 在桌面新建一个XX.desktop文件&#xff0c;以QtCreator为例。 &#xff08;注意这里不能使用sudo&#xff0c;因为这样会把文件的权限归为root&#xff0c;导致后续设置可执行程序不方便&#xff09; gedit qtcreator.desktop在XX.desktop文件中…

Chiplet,汽车“芯”风向

异构集成、高速互联、算力灵活可扩展正在成为新一轮汽车芯片竞争的焦点。尤其是随着以ChatGPT为代表的大数据、大模型产品在车端的落地&#xff0c;对于芯片的要求还在持续提升。 本周&#xff0c;12家日本汽车制造商&#xff08;包括丰田、日产、本田等&#xff09;、零部件制…

[技术杂谈]nvidia-smi参数和显示信息解释

GPU&#xff1a;本机中的GPU编号&#xff0c;从0开始&#xff0c;上图为0&#xff0c;一块GPU Fan&#xff1a;风扇转速&#xff08;0%-100%&#xff09;&#xff0c;N/A表示没有风扇 Name&#xff1a;GPU名字/类型&#xff0c;上图为NVIDIA GeForce . . . Temp&#xff1a;GPU…

《Numpy 简易速速上手小册》第10章:Numpy案例研究和实践技巧(2024 最新版)

文章目录 10.1 实际案例分析10.1.1 基础知识10.1.2 完整案例&#xff1a;天气数据分析10.1.3 拓展案例 1&#xff1a;股票价格分析10.1.4 拓展案例 2&#xff1a;信号处理 10.2 Numpy 最佳实践10.2.1 基础知识10.2.2 完整案例&#xff1a;高效数组操作10.2.3 拓展案例 1&#x…

2023年博客总结反思与未来规划

前言&#xff1a; 24将至&#xff0c;23收尾&#xff0c;作为一名电信专业的大一学生&#xff0c;我在这后半年学习了不少的编程知识&#xff0c;也写了几十篇博客&#xff0c;今天想反思自己在博客创作和知识学习中的不足并且对未来进行规划。 种下一棵树最好的时间是10年前&…

Doris 与 Clickhouse 对比(一)

1. 常用引擎 ☕️ Doris 表数据模型 duplicate key &#x1f3ac; 场景&#xff1a;适用于数据无需提前聚合的分析业务。 ⚠️ 注意点&#xff1a;只指定排序列&#xff0c;相同的行并不会合并。 unique key &#x1f3ac; 场景&#xff1a;适用于有更新需求的业务。 ⚠…

心灵鸡汤美文:温暖你的每一寸心田

1.人生就像一杯茶&#xff0c;不会苦一辈子&#xff0c;但总会苦一阵子。只有经历过苦涩&#xff0c;才能品味到甜美的滋味。 2.每一次失败都是一次宝贵的经验&#xff0c;它教会我们如何更好地面对困难和挑战。不要害怕失败&#xff0c;因为失败是成功的前奏。 3.人生最重要的…

开源项目MessageNest打造个性化消息推送平台多种通知方式

今天介绍一个开源项目&#xff0c;Message Nest - 可以打造个性化消息推送平台&#xff0c;整合邮件、钉钉、企业微信等多种通知方式。定制你的消息&#xff0c;让通知方式更灵活多样。 开源地址&#xff1a; https://github.com/engigu/Message-Push-Nest 测试平台 系统&am…

工业级数据采集通用网关在生产过程中的实际应用及其解决的问题-天拓四方

在工业生产过程中&#xff0c;实时数据是评估设备性能、监控生产流程、预测潜在问题以及优化资源配置的关键。数据采集不仅有助于提高生产效率&#xff0c;降低故障率&#xff0c;还可以为企业的决策提供科学依据。因此&#xff0c;选择一款高效、稳定的数据采集网关是至关重要…

使用nginx对视频、音频、图片等静态资源网址,加token签权

目前很多静态资源&#xff0c;都可以无权限验证&#xff0c;进行访问或转发&#xff0c;对有价值的资源进行签权&#xff0c;限制转发无法在代码中实现拦截&#xff0c;我们可以使用nginx对视频、音频、图片等静态资源网址&#xff0c;加token签权 如&#xff1a; http://192…

2024年新提出的算法:(凤头豪猪优化器)冠豪猪优化算法Crested Porcupine Optimizer(附Matlab代码)

本次介绍一种新的自然启发式元启发式算法——凤头豪猪优化器(Crested Porcupine Optimizer&#xff0c;CPO)。该成果于2024年1月发表在中科院1区SCI top期刊Knowledge-Based Systems&#xff08;IF 8.8&#xff09;上。 1、简介 受到凤头豪猪&#xff08;CP&#xff09;各种…

将 Quartz.NET 调度框架与 Stimulsoft Reports 结合使用

今天&#xff0c;我们将深入探讨软件开发的一种现代趋势 - 流程自动化&#xff0c;这自然是 Stimulsoft 产品中报表处理的一部分。在本文中&#xff0c;我们将讨论如何使用第三方调度程序自动执行与 Web 项目中的报告相关的任务。作为对报告执行操作的示例&#xff0c;我们考虑…

Hive 行列转换

行列转换 列转行 使用 lateral view explode(array|map) 或 lateral view inline(array_struct) 可以将列转换为行。 单列转多行&#xff0c;降维&#xff08;单列数组或键值对&#xff09; 示例1&#xff1a;explode(array(…)) select ..., A from T lateral view exp…