特殊类设计[下] --- 单例模式

文章目录

  • 5.只能创建一个对象的类
    • 5.1设计模式[2.5 万字详解:23 种设计模式](https://zhuanlan.zhihu.com/p/433152245)
    • 5.2单例模式
      • 1.饿汉模式
      • 1.懒汉模式
  • 6.饿汉模式
  • 7.懒汉模式
    • 7.1饿汉模式优缺点:
    • 7.2懒汉模式
      • 1.线程安全问题
      • 2.单例对象的析构问题
  • 8.整体代码
  • 9.C++11后可用的单例模式

5.只能创建一个对象的类

5.1设计模式2.5 万字详解:23 种设计模式

在这里插入图片描述

设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类的、代码设计经验的总结

为什么会产生设计模式这样的东西呢?

就像人类历史发展会产生兵法。最开始部落之间打仗时都是人拼人的对砍。春秋战国时期,七国之间经常打仗,发现打仗也是有套路的,孙子就总结出了《孙子兵法》

使用设计模式的目的是什么呢?

代码可重用性、代码更容易被理解、代码可靠性、代码编写工程化

设计模式是软件工程的基石脉络,如同大厦的结构一样

5.2单例模式

在这里插入图片描述

一个类只能创建一个对象,即单例模式

该模式可以保证系统中(一个进程)该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。[在此进程全局只有唯一一个 且 在任意地方可访问]

应用场景

在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,服务进程中的其他对象通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理

1.饿汉模式

1.懒汉模式

6.饿汉模式

程序启动时(main函数之前)就创建一个唯一的实例对象

class Singleton
{
public:static Singleton* GetPtrAtOnly(){return _ponly;}void PushData(const string& str){_mtx.lock();_vec.push_back(str);_mtx.unlock();}void Display(){_mtx.lock();for (auto& e : _vec){cout << e << endl;}cout << endl;_mtx.unlock();}private://构造函数私有化 -- 禁止类外创建对象Singleton(){}private:mutex _mtx;vector<string> _vec;//_ponly是一个存在于静态区的指针变量//这个指针初始化指向 一个Singleton对象static Singleton* _ponly;
};//在程序入口之前就完成单例对象的初始化
//类内声明 类外初始化
Singleton* Singleton::_ponly = new Singleton;int main()
{//Singleton s1;//static Singleton s2;//Singleton* p = new Singleton;Singleton::GetPtrAtOnly()->PushData("彭于晏");Singleton::GetPtrAtOnly()->PushData("吴彦祖");Singleton::GetPtrAtOnly()->PushData("黎明");Singleton::GetPtrAtOnly()->PushData("郭富城");Singleton::GetPtrAtOnly()->Display();return 0;
}

在这里插入图片描述
在这里插入图片描述

多线程单例模式之饿汉模式测试

int main()
{srand(time(0));int n = 10;thread t1([n](){for (size_t i = 0; i < n; ++i){Singleton::GetPtrAtOnly()->PushData("线程1: " + to_string(rand()));}});thread t2([n](){for (size_t i = 0; i < n; ++i){Singleton::GetPtrAtOnly()->PushData("线程2: " + to_string(rand()));}});t1.join();t2.join();Singleton::GetPtrAtOnly()->Display();return 0;
}

在这里插入图片描述

7.懒汉模式

7.1饿汉模式优缺点:

优点:相对懒汉模式而言简单一些
缺点:

  1. 影响进程启动速度
    饿汉模式main函数之前就要创建对象
    若单例对象初始化很慢(初始化操作很多[读取配置文件]) 对象1暂时不占用资源
    但是会影响后续程序的启动速度
  2. 多个单例类对象 实例启动顺序不确定
    两个有依赖关系的单例都是饿汉时
    若要求创建顺序:单例1--单例2
    饿汉模式无法控制顺序

7.2懒汉模式

1.线程安全问题

  1. 懒汉模式
static Singleton* GetPtrAtOnly()
{if (_ponly == nullptr){if (_ponly == nullptr){_ponly = new Singleton;}}return _ponly;
}

假设两个线程 线程1的对象实例化后进行了添加数据 此时线程2执行 覆盖线程1

  1. 饿汉模式不用考虑

线程在main函数后进行 饿汉模式在main函数前就创建了对象

    static Singleton* GetPtrAtOnly(){return _ponly;}Singleton* Singleton::_ponly = new Singleton;

加锁保护

	static Singleton* GetPtrAtOnly(){_imtx.lock();if (_ponly == nullptr){_ponly = new Singleton;}_imtx.unlock();return _ponly;}

每次创建对象都要 加锁解锁 有无改进办法?

	static Singleton* GetPtrAtOnly(){if (_ponly == nullptr){_imtx.lock();_ponly = new Singleton;_imtx.unlock();}return _ponly;}

此时相当于没加锁 跟没加锁造成的问题一样 以下的双检查加锁才是解决办法

	static Singleton* GetPtrAtOnly(){//懒汉模式 不在外部加锁 提高效率 -- 要不然每次创建对象都要加锁if (_ponly == nullptr){_imtx.lock();//线程安全 t1判断为空 new对象 t2来了不为空 不再new 更正了覆盖问问题if (_ponly == nullptr){_ponly = new Singleton;}_imtx.unlock();}return _ponly;}

2.单例对象的析构问题

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

8.整体代码

#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <list>
#include <vector>
#include <algorithm>
#include <array>
#include <time.h>
#include <queue>
#include <stack>
#include <string>
#include <set>
#include <map>
#include <unordered_set>
#include <unordered_map> 
#include <thread> 
#include <functional>
#include <assert.h>
#include<mutex>
using namespace std;// 饿汉模式:
/*
class Singleton
{
public:static Singleton* GetPtrAtOnly(){return _ponly;}void PushData(const string& str){_mtx.lock();_vec.push_back(str);_mtx.unlock();}void Display(){_mtx.lock();for (auto& e : _vec){cout << e << endl;}cout << endl;_mtx.unlock();}private://构造函数私有化 -- 禁止类外创建对象Singleton(){}private:mutex _mtx;vector<string> _vec;//_ponly是一个存在于静态区的指针变量//这个指针初始化指向 一个Singleton对象//这里可以直接static Singleton _only; 他是一个对象 程序结束时调用析构//而懒汉模式只能是指针因为他要判断是否空再去创建对象//所以懒汉模式不得不写一个对象回收实现自动析构static Singleton* _ponly;
};
Singleton* Singleton::_ponly = new Singleton;
*///懒汉模式:第一次访问实例对象时[第一次调用GetPtrAtOnly()]创建
class Singleton
{
public://获取单例对象static Singleton* GetPtrAtOnly(){if (_ponly == nullptr){_ptrmtx.lock();if (_ponly == nullptr){_ponly = new Singleton;}_ptrmtx.unlock();}return _ponly;}// 一般全局都要使用单例对象 // 所以单例对象一般不需要显示释放  // 特殊场景 -- 显示释放//释放单例对象static void DeletePtrAtOnly(){_ptrmtx.lock();if (_ponly != nullptr){delete _ponly;_ponly = nullptr;}_ptrmtx.unlock();}void PushData(const string& str){_vecmtx.lock();_vec.push_back(str);_vecmtx.unlock();}void Display(){_vecmtx.lock();for (auto& e : _vec){cout << e << endl;}cout << endl;_vecmtx.unlock();}~Singleton(){// 要求程序结束时// 将数据写到文件 // 单例对象析构时[持久化]// 即析构前做事情 // 写文件操作//DeletePtrAtOnly();//存在一种情况 写文件操作代码量太大 最后忘记调用DeletePtrAtOnly();//此时有没有析构单例对象 怎么办? 能不能搞得智能一点?//类比智能指针 再搞一个类 使得实现"自动化"//_gc是一个静态局部变量 他的析构发生在main函数结束后 程序结束时//_gc析构时 会调用他的析构函数~Garbage_Collection(); //他的析构时会调用单例对象的析构函数 由此实现自动化}// 单例对象回收class Garbage_Collection{public:~Garbage_Collection(){DeletePtrAtOnly();}};static Garbage_Collection _gc;private:Singleton(){}//有锁时 不禁用拷贝构造也行 因为锁使得vector不能push_backSingleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;
private:mutex _vecmtx;vector<string> _vec;static mutex _ptrmtx;static Singleton* _ponly;
};mutex Singleton::_ptrmtx;Singleton* Singleton::_ponly = nullptr;Singleton::Garbage_Collection Singleton::_gc;int main()
{//Singleton s(*Singleton::GetPtrAtOnly());srand(time(0));int n = 20;thread t1([n](){for (size_t i = 0; i < n; ++i){Singleton::GetPtrAtOnly()->PushData("线程1: " + to_string(rand()));}});thread t2([n](){for (size_t i = 0; i < n; ++i){Singleton::GetPtrAtOnly()->PushData("线程2: " + to_string(rand()));}});t1.join();t2.join();Singleton::GetPtrAtOnly()->Display();return 0;
}

9.C++11后可用的单例模式

C++11单例模式简单写法:将对象定义GetPtrAtOnly()函数的局部静态变量 返回对象的引用 在GetPtrAtOnly()函数首次调用时完成静态对象初始化
当某一个线程调用GetPtrAtOnly()执行初始化静态变量时,若其他线程正在执行初始化该静态变量 则先初始化上一进程

class Singleton
{
public:// C++11后才可以保证初始化静态对象的线程安全问题static Singleton* GetPtrAtOnly(){static Singleton one; return &one;}void PushData(const string& str){_vecmtx.lock();_vec.push_back(str);_vecmtx.unlock();}void Display(){_vecmtx.lock();for (auto& e : _vec){cout << e << endl;}cout << endl;_vecmtx.unlock();}~Singleton(){}
private:Singleton(){}Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;mutex _vecmtx;vector<string> _vec;
};

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

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

相关文章

UVa129 Krypton Factor(困难的串)

1、题目 2、题意 如果一个字符串包含两个相邻的重复子串&#xff0c;则称它是“容易的串”&#xff0c;其他串称为“困难的串”。例如&#xff0c;BB、ABCDACABCAB、ABCDABCD都是容易的的串&#xff0c;而D、DC、ABDAB、CBABCBA 都是困难的串。 输入正整数 k k k 和 L L L&a…

Linux虚拟机部署与发布项目(Windows版本)

目录 前言 一、虚拟机部署项目的流程 二、单机项目 1. 本机测试 2.虚拟机部署项目 三、前后端分离项目 前言 在软件开发过程中&#xff0c;部署和发布项目是非常重要的一环。使用虚拟机技术可以方便、灵活且可重复使用地部署和发布项目。本篇博客将介绍如何在 Windows 环…

【密评】商用密码应用安全性评估从业人员考核题库(十四)

商用密码应用安全性评估从业人员考核题库&#xff08;十四&#xff09; 国密局给的参考题库5000道只是基础题&#xff0c;后续更新完5000还会继续更其他高质量题库&#xff0c;持续学习&#xff0c;共同进步。 3251 单项选择题 根据GB/T 39786《信息安全技术 信息系统密码应用…

Go基础——基础语法

1、简介 Go&#xff08;又称Golang&#xff09;是Google开发的一种静态强类型、编译型、并发型&#xff0c;并具有垃圾回收功能的编程语言。语法类似于C&#xff0c;专为高性能和并发程序而设计。通常用于网络编程、云服务器、游戏服务器、DevOps、Web应用、分布式系统、容器虚…

mysql-linux归档版安装

什么是归档版安装&#xff1f;简单来说就是编译好的软件压缩打包版。 说明&#xff1a;我这里服务器之前已经装过一个不同版本的mysql&#xff0c;已经占用了3306端口&#xff0c;所以这里我用3307端口来演示&#xff0c;命令和官方的稍有不同&#xff0c;不过步骤都是差不多的…

搜索引擎搜索技巧总结

晚上在B站上刷到一个关于搜索技巧的干货视频&#xff0c;这个视频真的不错&#xff0c;结尾还提到了AI时代的搜索思路之前自己也零碎的探索出了一些搜索技巧&#xff0c;但是没有总结&#xff0c;就没法稳定的加入自己的工作流&#xff0c;持续提高效率受到这个视频的启发&…

计算机视觉 激光雷达结合无监督学习进行物体检测的工作原理

一、简述 激光雷达是目前正在改变世界的传感器。它集成在自动驾驶汽车、自主无人机、机器人、卫星、火箭等中。该传感器使用激光束了解世界,并测量激光击中目标返回所需的时间,输出是点云信息,利用这些信息,我们可以从3D点云中查找障碍物。 从自动驾驶汽车的角度看激光雷达…

专业135总分400+西安交通大学信息与通信工程学院909/815考研经验分享

今年初试发挥不错&#xff0c;400&#xff0c;专业课135&#xff0c;将近一年复习一路走来&#xff0c;感慨很多&#xff0c;希望以下经历可以给后来的同学提供一些参考。 初试备考经验 公共课&#xff1a;三门公共课&#xff0c;政治&#xff0c;英语&#xff0c;数学。在备考…

轮转数组(Java)

大家好我是苏麟 , 这篇文章是凑数的 ... 轮转数组 描述 : 给定一个整数数组 nums&#xff0c;将数组中的元素向右轮转 k 个位置&#xff0c;其中 k 是非负数。 题目 : 牛客 NC110 旋转数组: 这里牛客给出了数组长度我们直接用就可以了 . LeetCode 189.轮转数组 : 189. 轮…

Nginx 的配置文件(负载均衡,反向代理)

Nginx可以配置代理多台服务器&#xff0c;当一台服务器宕机之后&#xff0c;仍能保持系统可用。 cmd查找端口是否使用&#xff1a;netstat -ano Nginx出现403 forbidden #解决办法&#xff1a;修改web目录的读写权限&#xff0c;或者是把nginx的启动用户改成目录的所属用户&…

田忌赛马(贪心算法)

分为两种情况&#xff1a; 1.田最快 快于 王最快&#xff08;田最快面对任何都赢&#xff09; ->故和王最快比&#xff0c;消耗王最快 2.田最快 小于等于 王最快&#xff08;王最快面对任何都可赢&#xff09; 则比最慢&#xff08;意在找一匹无法赢任何马的马&#…

20.1 OpenSSL 字符BASE64压缩算法

OpenSSL 是一种开源的加密库&#xff0c;提供了一组用于加密和解密数据、验证数字证书以及实现各种安全协议的函数和工具。它可以用于创建和管理公钥和私钥、数字证书和其他安全凭据&#xff0c;还支持SSL/TLS、SSH、S/MIME、PKCS等常见的加密协议和标准。 OpenSSL 的功能非常…

shell实验

1&#xff0e;编写脚本for1.sh&#xff0c;使用for循环创建20账户&#xff0c;账户名前缀由用户从键盘输入&#xff0c;账户初始密码由用户输入&#xff0c;例如&#xff1a;test1、test2、test3、....、test10 编写脚本&#xff0c;使用read -p提醒用户从键盘输入账户名前缀以…

[论文笔记]GTE

引言 今天带来今年的一篇文本嵌入论文GTE, 中文题目是 多阶段对比学习的通用文本嵌入。 作者提出了GTE,一个使用对阶段对比学习的通用文本嵌入。使用对比学习在多个来源的混合数据集上训练了一个统一的文本嵌入模型,通过在无监督预训练阶段和有监督微调阶段显著增加训练数…

hdlbits系列verilog解答(移位寄存器)-23

文章目录 一、问题描述二、verilog源码三、仿真结果 一、问题描述 您将获得一个具有两个输入和一个输出的模块 my_dff &#xff08;实现 D 触发器&#xff09;。实例化其中的三个&#xff0c;然后将它们链接在一起以形成长度为 3 的移位寄存器。端口 clk 需要连接到所有实例。…

三篇论文:速览GPT在网络安全最新论文中的应用案例

GPT在网络安全领域的应用案例 写在最前面论文1&#xff1a;Chatgpt/CodeX引入会话式 APR 范例利用验证反馈LLM 的长期上下文窗口&#xff1a;更智能的反馈机制、更有效的信息合并策略、更复杂的模型结构、鼓励生成多样性和GPT类似的步骤&#xff1a;Conversational APR 对话式A…

Windows VS C++工程:包含目录、库目录、附加依赖项、附加包含目录、附加库目录配置与静态库、动态库的调用——以OCCI的配置为例

文章目录 1 包含目录&#xff08;Include Directories&#xff09;/ 附加包含目录&#xff08;Additional Include Directories&#xff09;1.1 区别和作用1.2 设置路径 2 库目录&#xff08;Library Directories&#xff09;/ 附加库目录&#xff08;Additional Library Direc…

element ui el-table表格纵向横向滚动条去除并隐藏空白占位列

需求 当table内容列过多时&#xff0c;可通过height属性设置table高度以固定table高度、固定表头&#xff0c;使table内容可以滚动 现在需求是右侧滚动条不好看&#xff0c;需要去除滚动条&#xff0c;并隐藏滚动条所占列的位置 // ----------修改elementui表格的默认样式-…

Spring MVC的常用注解

目录 RequestMapping 例子&#xff1a; RequestMapping 支持什么类型的请求 使 RequestMapping 只支持特定的类型 RestController 通过 HTTP 请求传递参数给后端 1.传递单个参数 注意使⽤基本类型来接收参数的情况 2.传递多个参数 3.传递对象 4.RequestParam 后端参数…

配置Super-VLAN下的DHCP服务器示例

组网需求 如图1所示&#xff0c;某公司拥有两个部门&#xff0c;为了节省IP地址&#xff0c;部门A和部门B规划为同一网段&#xff1b;为了提升业务安全性&#xff0c;将不同部门的用户划分到不同VLAN中。企业管理员为了方便统一管理&#xff0c;希望部门内终端通过DHCP服务器动…