当前位置: 首页 > news >正文

c++模式之单例模式详解

c++模式之单例模式详解

  • 1.概念
  • 2.懒汉模式示例(缺点)
  • 3.懒汉模式线程安全
  • 4.饿汉式创建单例
  • 5.饿汉模式线程示例

1.概念

单例模式是指在整个系统生命周期内,保证一个类只能产生一个实例,确保该类的唯一性.
使用单例两个原因:
1.节省资源。一个类只有一个实例,不存在多份实例,节省资源。
2.方便控制。在一些操作公共资源的场景时,避免了多个对象引起的复杂操作
单例类的特点

  • 构造函数和析构函数为私有类型,目的是禁止外部构造和析构。
  • 拷贝构造函数和赋值构造函数是私有类型,目的是禁止外部拷贝和赋值,确保实例的唯一性。
  • 类中有一个获取实例的静态方法,可以全局访问。

2.懒汉模式示例(缺点)

getInstance函数使用了懒汉式单例模式的实现方式。它首先检查静态成员变量instance是否为空,如果为空则创建一个新的实例,否则直接返回已有的实例。这种实现方式在单线程环境下是有效的,但在多线程环境下可能会导致线程安全问题。
在多线程环境下,多个线程可能会同时检查到instance为空,然后同时创建多个实例,违背了单例模式的初衷。为了解决这个问题,我们需要在创建实例时添加同步机制,以确保只有一个线程能够创建实例。

#include <iostream>
#include <mutex>
#include <ctime>
#include <vector>
using namespace std;void sleep(int time) { clock_t head = clock(); while (clock() - head <= time) {} }class Singleton {
private:static Singleton* instance; // 静态成员变量,用于保存单例实例Singleton() {} // 私有构造函数,防止外部实例化public:static Singleton* getInstance() {if (instance == nullptr) {instance = new Singleton();}return instance;}void someFunction() {// 单例的其他成员函数cout << "HELLO WORLD" << endl;}
};Singleton* Singleton::instance = nullptr; // 初始化静态成员变量int main() {Singleton* obj1 = Singleton::getInstance();obj1->someFunction();Singleton* obj2 = Singleton::getInstance();obj2->someFunction();// obj1和obj2是同一个实例return 0;
}

3.懒汉模式线程安全

添加了一个静态成员变量mutex作为互斥锁,用于线程同步。在getInstance函数中,我们首先进行一次非线程安全的检查,如果instance为空,才会获取互斥锁并再次检查instance是否为空。这样可以确保只有一个线程能够创建实例。
使用了std::lock_guard来自动管理锁的加锁和解锁,以避免手动处理锁的释放。这样可以确保在任何情况下,无论是正常返回还是发生异常,都会自动释放锁。
这个改进后的示例提供了一种线程安全的懒汉式单例模式实现方式,可以在多线程环境下正常工作

#include <iostream>
#include <mutex>
using namespace std;class Singleton {
private:static Singleton* instance; // 静态成员变量,用于保存单例实例static std::mutex mutex; // 互斥锁,用于线程同步Singleton() {} // 私有构造函数,防止外部实例化public:static Singleton* getInstance() {if (instance == nullptr) {lock_guard<std::mutex> lock(mutex); // 加锁if (instance == nullptr) {instance = new Singleton();}}return instance;}void someFunction() {// 单例的其他成员函数cout << "mutex hello world" << endl;}
};Singleton* Singleton::instance = nullptr; // 初始化静态成员变量
std::mutex Singleton::mutex; // 初始化互斥锁int main() {Singleton* obj1 = Singleton::getInstance();obj1->someFunction();Singleton* obj2 = Singleton::getInstance();obj2->someFunction();// obj1和obj2是同一个实例return 0;
}

4.饿汉式创建单例

它是一种在程序启动时就创建实例的单例模式。下面是一个简单的示例来说明如何实现C++的饿汉式单例模式。

Singleton类使用饿汉式的方式创建实例。在静态成员变量instance的定义处,我们直接使用new操作符创建了一个Singleton的实例,并将其赋值给instance。这样,在程序启动时,实例就会被创建并初始化。
在getInstance函数中,我们直接返回已经创建好的实例,而无需再进行实例化。
在main函数中,我们通过调用Singleton::getInstance()来获取单例实例,并调用其成员函数。由于使用了饿汉式创建实例,obj1和obj2实际上是同一个实例。
饿汉式在程序启动时就创建了实例,因此会占用一定的内存空间。此外,如果实例的创建过程较为复杂或耗时,可能会影响程序的启动速度。因此,在选择单例模式的实现方式时,需要根据具体的需求和场景来决定使用懒汉式还是饿汉式。

#include <iostream>
using namespace std;class Singleton {
private:static Singleton* instance; // 静态成员变量,用于保存单例实例Singleton() {} // 私有构造函数,防止外部实例化public:static Singleton* getInstance() {return instance;}void someFunction() {cout << "Hungry Han style instance" << endl;}
};Singleton* Singleton::instance = new Singleton(); // 在静态成员变量初始化时创建实例int main() {Singleton* obj1 = Singleton::getInstance();obj1->someFunction();Singleton* obj2 = Singleton::getInstance();obj2->someFunction();// obj1和obj2是同一个实例return 0;
}

5.饿汉模式线程示例

#include <iostream>
#include <thread>
using namespace std;class ThreadSingleton {
private:static ThreadSingleton* instance; // 静态成员变量,用于保存单例实例ThreadSingleton() {} // 私有构造函数,防止外部实例化public:static ThreadSingleton* getInstance() {return instance;}void calculateSquareArea(double side) {double area = side * side;std::cout << "The area of the square is: " << area << std::endl;}
};ThreadSingleton* ThreadSingleton::instance = new ThreadSingleton(); // 在静态成员变量初始化时创建实例int main() {std::thread t1([&]() {ThreadSingleton* obj1 = ThreadSingleton::getInstance();obj1->calculateSquareArea(5);});std::thread t2([&]() {ThreadSingleton* obj2 = ThreadSingleton::getInstance();obj2->calculateSquareArea(8);});t1.join();t2.join();return 0;
}

相关文章:

c++模式之单例模式详解

c模式之单例模式详解 1.概念2.懒汉模式示例&#xff08;缺点&#xff09;3.懒汉模式线程安全4.饿汉式创建单例5.饿汉模式线程示例 1.概念 单例模式是指在整个系统生命周期内&#xff0c;保证一个类只能产生一个实例&#xff0c;确保该类的唯一性. 使用单例两个原因&#xff1a…...

【gpts】学算法题[缺失的第一个正数](https://leetcode.cn/problems/first-missing-positive/)

给出你的题解 (https://leetcode.cn/problems/first-missing-positive/) public class Solution { public int firstMissingPositive(int[] nums) {int len nums.length;for (int i 0; i < len; i) {while (nums[i] > 0 && nums[i] < len && nums…...

车牌识别 支持12种中文车牌类型 车牌数据集下载

开源代码 如果觉得有用&#xff0c;不妨给个Star⭐️&#x1f31f;支持一下吧~ 谢谢&#xff01; Acknowledgments & Contact 1.WeChat ID: cbp931126 2.QQ Group&#xff1a;517671804 加微信(备注&#xff1a;PlateAlgorithm),进讨论群可以获得10G大小的车牌检测和识…...

Servlet---上传文件

文章目录 上传文件的方法上传文件的示例前端代码示例后端代码示例 上传文件的方法 上传文件的示例 前端代码示例 <body><form action"upload" method"post" enctype"multipart/form-data"><input type"file" name&qu…...

量子计算+物流!“最后一英里”配送难题Unisys成功实时决策

&#xff08;图片来源&#xff1a;网络&#xff09; 此前&#xff0c;供应链行业一直致力于手工操作&#xff0c;严重依赖于纸质系统。后来随着客户需求的不断变化&#xff0c;这种传统方法逐渐显出不足之处。供应链行业正在迅速转向现代化&#xff0c;采用自动化和数据驱动的…...

2023年【四川省安全员A证】复审考试及四川省安全员A证考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 四川省安全员A证复审考试根据新四川省安全员A证考试大纲要求&#xff0c;安全生产模拟考试一点通将四川省安全员A证模拟考试试题进行汇编&#xff0c;组成一套四川省安全员A证全真模拟考试试题&#xff0c;学员可通过…...

C++刷题 -- 二分查找

C刷题 – 二分查找 文章目录 C刷题 -- 二分查找一、原理二、例题1.二分查找2.使用二分查找确定target左右边界3.x的平方根 一、原理 条件&#xff1a;数组为有序数组&#xff0c;数组中无重复元素&#xff0c;因为一旦有重复元素&#xff0c;使用二分查找法返回的元素下标可能…...

PHPmail 发送邮件错误 550 的原因是什么?

电子邮件错误消息链接到简单邮件传输协议 (SMTP)&#xff0c;这是一组发送和接收电子邮件的标准化规则。因此&#xff0c;它也称为 SMTP 550 错误代码。在某些情况下&#xff0c;电子邮件错误 550 是由收件人一方的问题引起的。 以下是电子邮件错误 550 的一些可能原因&#x…...

数字化转型导师坚鹏:数字化时代银行网点厅堂营销5大难点分析

数字化时代银行网点厅堂营销存在以下5大难点&#xff1a; 1、识别难。识别有效的客户比较难&#xff0c;传统的厅堂识别主要依据客户的衣着气质等主管感受&#xff0c;判断客户是否为潜在中高端客户&#xff0c;提供相关服务。大堂经理主管识别与智能化系统识别相结合&#xf…...

www.testfire.nets渗透测试报告

www.testfire.nets渗透测试报告 一、测试综述 1.1.测试⽬的 通过实施针对性的渗透测试&#xff0c;发现testfire.net⽹站的安全漏洞&#xff0c;锻炼自己的渗透水平 1.2.测试范围 域名&#xff1a;www.testfire.net IP:65.61.137.117 测试时间&#xff1a; 2023年11月…...

多模态大一统:通向全模态学习和通用人工智能的未来之路

随着AI技术的不断发展&#xff0c;研究者们正试图构建一种真正通用的人工智能&#xff0c;它能像人们那样以统一的方式处理和理解多种模态的信息。多模态大一统是这一愿景的关键&#xff0c;它旨在开启全模态LLM&#xff08;深度学习语言模型&#xff09;和通用AI时代的大门。在…...

实用篇-ES-DSL查询文档

数据的存储不是目的&#xff0c;我们希望从海量的酒店数据中检索出需要的信息&#xff0c;这就是ES的搜索功能 官方文档: https://elastic.co/guide/en/elasticsearch/reference/current/query-dsl.html#query-dsl。DSL是用来查询文档的 Elasticsearch提供了基于JSON的DSL来定…...

Nacos配置管理

将配置交给Nacos管理的步骤 1、在Nacos中添加配置文件 2、在微服务中引入nacos的config依赖 3、在微服务中添加bootstrap.yml&#xff0c;配置nacos地址、当前环境、服务名称、文件后缀名。这些决定了程序启动时去nacos读取哪个文件 Nacos配置更改后&#xff0c;微服务可以实…...

【前端学java】Java中的异常处理(15)完结

往期回顾&#xff1a; 【前端学java】JAVA开发的依赖安装与环境配置 &#xff08;0&#xff09;【前端学java】java的基础语法&#xff08;1&#xff09;【前端学java】JAVA中的packge与import&#xff08;2&#xff09;【前端学java】面向对象编程基础-类的使用 &#xff08;…...

深入理解MySQL存储引擎、InnoDB与MyISAM的比较以及事务处理机制

介绍 MySQL是一款强大而灵活的关系型数据库管理系统&#xff0c;它支持多种存储引擎&#xff0c;每个引擎都有其独特的特点和适用场景。在本篇博客中&#xff0c;我们将深入探讨MySQL存储引擎的种类、InnoDB与MyISAM的区别&#xff0c;以及事务的概念及其在MySQL中的实现方式。…...

webpack 中,filename 和 chunkFilename 的区别

filename filename 是一个很常见的配置&#xff0c;就是对应于 entry 里面的输入文件&#xff0c;经过webpack打包后输出文件的文件名。比如说经过下面的配置&#xff0c;生成出来的文件名为 index.min.js。 chunkFilename chunkFilename 指未被列在 entry 中&#xff0c;却…...

gitlab 实战

一.安装依赖 yum install -y curl policycoreutils-python openssh-server perl 二.安装gitlab yum install gitlab-jh-16.0.3-jh.0.el7.x86_64.rpm 三.修改下面的 vim /etc/gitlab/gitlab.rbexternal_url http://192.168.249.156 四.初始化 gitlab-ctl reconfigure 五.查看状…...

openGauss学习笔记-128 openGauss 数据库管理-设置透明数据加密(TDE)

文章目录 openGauss学习笔记-128 openGauss 数据库管理-设置透明数据加密&#xff08;TDE&#xff09;128.1 概述128.2 前提条件128.3 背景信息128.4 密钥管理机制128.5 表级加密方案128.6 创建加密表128.7 切换加密表加密开关128.8 对加密表进行密钥轮转 openGauss学习笔记-12…...

Redis从入门到精通(三)-高阶篇

文章目录 0. 前言[【高阶篇】3.1 Redis协议(RESP )详解](https://blog.csdn.net/wangshuai6707/article/details/132742584)[【高阶篇】3.3 Redis之底层数据结构简单动态字符串(SDS)详解](https://blog.csdn.net/wangshuai6707/article/details/131101404)[【高阶篇】3.4 Redis…...

线性表--队列-1

文章目录 主要内容一.队列基础练习题1.用链式存储方式的队列进行删除操作时需要 ( D ).代码如下&#xff08;示例&#xff09;: 2.若以1,2,3,4作为双端队列的输入序列&#xff0c;则既不能由输入受限的双端队列得到&#xff0c;又不能由输出受限的双端队列得到的输出序列是( C …...

CANN/cann-bench GRU算子评测

GRU 算子 API 描述 【免费下载链接】cann-bench 评测AI在处理CANN领域代码任务的能力&#xff0c;涵盖算子生成、算子优化等领域&#xff0c;支撑模型选型、训练效果评估&#xff0c;统一量化评估标准&#xff0c;识别Agent能力短板&#xff0c;构建CANN领域评测平台&#xff0…...

基于Electron与LLM的CK3智能对话模组开发实战

1. 项目概述&#xff1a;当《十字军之王3》的宫廷角色开始“思考”如果你和我一样&#xff0c;是个策略游戏迷&#xff0c;同时又对AI技术充满好奇&#xff0c;那么“Voices of the Court”&#xff08;宫廷之声&#xff09;这个项目绝对会让你眼前一亮。简单来说&#xff0c;这…...

CANN/driver获取能力组信息API

dcmi_get_capability_group_info 【免费下载链接】driver 本项目是CANN提供的驱动模块&#xff0c;实现基础驱动和资源管理及调度等功能&#xff0c;使能昇腾芯片。 项目地址: https://gitcode.com/cann/driver 函数原型 int dcmi_get_capability_group_info(int card_…...

CANN/catlass Block Mmad基础模板

Block Mmad基础模板 【免费下载链接】catlass 本项目是CANN的算子模板库&#xff0c;提供NPU上高性能矩阵乘及其相关融合类算子模板样例。 项目地址: https://gitcode.com/cann/catlass 代码位置 [TOC] BlockMmad 功能说明 block层级mmad计算&#xff0c;非TLA实现&am…...

蜂鸟v2 E203:开源RISC-V MCU SoC架构解析与FPGA开发实战

1. 项目概述&#xff1a;从开源蜂鸟E203到蜂鸟v2 E203的演进如果你对RISC-V处理器设计、嵌入式系统开发&#xff0c;或者单纯想找一个能跑在FPGA上的、五脏俱全的开源MCU SoC来学习&#xff0c;那么“蜂鸟v2 E203”&#xff08;Hummingbirdv2 E203&#xff09;这个项目绝对值得…...

Openclaw-Setup:开源桌面自动化框架部署与实战指南

1. 项目概述与核心价值最近在折腾一个挺有意思的项目&#xff0c;叫“Openclaw-Setup”。光看这个名字&#xff0c;可能有点摸不着头脑&#xff0c;但如果你对自动化、机器人流程自动化&#xff08;RPA&#xff09;或者桌面应用控制感兴趣&#xff0c;那这个项目绝对值得你花时…...

SpringBoot配置中的变量引用技巧

在SpringBoot应用中,配置文件的灵活性是其一大优势。我们经常需要在配置文件中引用其他属性的值来动态生成新的配置项。本文将通过一个实例讲解如何在SpringBoot的application.yml文件中使用变量引用技术,特别是如何将一个变量的值作为Map的键名。 背景 假设我们有一个Spri…...

OpenSpeedy完全指南:免费开源游戏加速工具让游戏体验飞跃

OpenSpeedy完全指南&#xff1a;免费开源游戏加速工具让游戏体验飞跃 【免费下载链接】OpenSpeedy &#x1f3ae; An open-source game speed modifier. 项目地址: https://gitcode.com/gh_mirrors/op/OpenSpeedy OpenSpeedy是一款完全免费且开源的游戏加速工具&#xf…...

如何用AKShare快速搞定金融数据获取?终极实战指南

如何用AKShare快速搞定金融数据获取&#xff1f;终极实战指南 【免费下载链接】akshare AKShare is an elegant and simple financial data interface library for Python, built for human beings! 开源财经数据接口库 项目地址: https://gitcode.com/gh_mirrors/aks/akshar…...

终极指南:如何通过Machine-Learning-Tutorials实现模型决策透明化技术

终极指南&#xff1a;如何通过Machine-Learning-Tutorials实现模型决策透明化技术 【免费下载链接】Machine-Learning-Tutorials machine learning and deep learning tutorials, articles and other resources 项目地址: https://gitcode.com/gh_mirrors/ma/Machine-Learni…...