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

如何优雅地实现全局唯一?深入理解单例模式

如何优雅地实现全局唯一?深入理解单例模式

一、什么是单例模式?

单例模式是一种创建型设计模式,旨在确保一个类只有一个实例,并为该实例提供全局访问点,从而避免全局变量的命名污染,并支持延迟初始化Wikipedia。

关键点:

  • 私有构造函数(禁止外部new创建)
  • 静态私有实例变量
  • 静态公有获取方法
二、C++实现示例
#include<iostream>
using namespace std;
#if 0
// 饿汉模式 -> 定义类的时候创建单例对象
// 在多线程的场景下没用线程安全问题
// 线程安全:多线程同时访问单例模式
// 定义一个单例模式的任务队列
class TaskQueue
{
public:TaskQueue(const TaskQueue & t) = delete;TaskQueue& operator =(const TaskQueue& t) = delete;static TaskQueue *getInstance(){return m_taskQ;}void print(){cout<<"我是单例对象的一个成员函数..."<<endl;}
private:TaskQueue() = default;// 只能通过类名访问静态成员属性或方法static TaskQueue* m_taskQ;
};
TaskQueue* TaskQueue::m_taskQ = new TaskQueue;
#endif#if 1
// 懒汉模式 -> 什么时候使用这个单例,再使用的时候再去创建对应的实例
// 在多线程的场景下可能存在线程安全问题
// 加互斥锁,让线程依次访问单例对象
// 比较节省内存空间
class TaskQueue
{
public:TaskQueue(const TaskQueue & t) = delete;TaskQueue& operator =(const TaskQueue& t) = delete;static TaskQueue *getInstance(){if(m_taskQ == nullptr){m_taskQ = new TaskQueue;}return m_taskQ;}void print(){cout<<"我是单例对象的一个成员函数..."<<endl;}
private:TaskQueue() = default;//只能通过类名访问静态成员属性或方法static TaskQueue* m_taskQ;
};
TaskQueue* TaskQueue::m_taskQ = nullptr;
#endif
int main()
{TaskQueue* taskQ = TaskQueue::getInstance();taskQ->print();return 0;
}

懒汉模式使用双重检查锁定解决线程安全问题

问题原因:

多线程调用懒汉模式getInstance(),就会创建出多个TaskQueue的实例,违背单例模式的定义,所谓的单例就是只能有唯一的一个单例对象

解决方法:

1、互斥锁解决线程安全问题

互斥锁:避免同时访问,按顺序依次访问

使用原子变量解决双重检查的问题:

//互斥锁头文件
#include<mutex>
...mutex TaskQueue::m_mutex;
...static mutex m_mutex;
...static TaskQueue *getInstance(){if(m_taskQ == nullptr)//第一次检查{//进行加锁操作m_mutex.lock();if(m_taskQ == nullptr)//第二次检查{m_taskQ = new TaskQueue;}//进行解锁操作 *注意*一个程序加锁之后一定要解锁,否则导致死锁m_mutex.unlock();}return m_taskQ;} 
...

使用原子变量解决双重检查的问题

//原子变量头文件
#include<atomic>
...atomic<TaskQueue*> TaskQueue::m_taskQ = nullptr; //初始化
...static atomic<TaskQueue*>m_taskQ;
...static TaskQueue *getInstance(){TaskQueue* task = m_taskQ.load();if(task == nullptr){m_mutex.lock();task = m_taskQ.load();//通过原子变量加载实例化指针if(task == nullptr){task = new TaskQueue;m_taskQ.store(task);}m_mutex.unlock();}return task;} 
...

2、局部静态对象解决线程安全问题

// 使用静态的局部对象解决线程安全问题 ->编译器支持C++11
...
class TaskQueue
{
public:TaskQueue(const TaskQueue & t) = delete;TaskQueue& operator =(const TaskQueue& t) = delete;static TaskQueue *getInstance(){static TaskQueue task;return &task;}void print(){cout<<"我是单例对象的一个成员函数..."<<endl;}
private:TaskQueue() = default;};
...
http://www.xdnf.cn/news/46387.html

相关文章:

  • uniapp微信小程序实现sse
  • 深度学习优化器详解:SGD、Adam与AdamW
  • C#/.NET/.NET Core技术前沿周刊 | 第 35 期(2025年4.14-4.20)
  • docker 安装 MySQL
  • 【Oracle专栏】函数中SQL拼接参数 报错处理
  • 【网络原理】TCP协议如何实现可靠传输(确认应答和超时重传机制)
  • Vue3 + TypeScript,关于item[key]的报错处理方法
  • Cherry Studio配置MCP服务全流程解析
  • AIGC通信架构深度优化指南
  • C++在VR/AR图形处理开发中的实战应用
  • 02【初体验】安装、配置与 Hello Cargo:踏出 Rust 开发第一步
  • Lora 微调自定义device_map
  • 【Linux】Rhcsa复习5
  • 阿里云 dataworks maxcompute创建python脚本实现列转行 脚本demo示例。
  • 06 GE Modifier
  • AUTOSAR图解==>AUTOSAR_RS_BSWModuleDescriptionTemplate
  • 19. git reflog
  • 力扣每日打卡16 781. 森林中的兔子(中等)
  • C++项目 —— 基于多设计模式下的同步异步日志系统(4)(双缓冲区异步任务处理器(AsyncLooper)设计)
  • 家庭电脑隐身后台自动截屏软件,可远程查看
  • Spring Data MongoDB 精华:给新手的核心注解指南
  • 从内核到用户态:Linux信号内核结构、保存与处理全链路剖析
  • 图论基础:图存+记忆化搜索
  • 基于论文的大模型应用:基于SmartETL的arXiv论文数据接入与预处理(三)
  • 嵌入式---零点漂移(Zero Drift)
  • go+mysql+cocos实现游戏搭建
  • jetpack之LiveData的原理解析
  • 【25软考网工】第二章(8)差错控制、奇偶校验、CRC、海明码
  • Doris + Iceberg 构建冷热分层数据湖架构:架构设计与实战指南
  • Linux驱动开发--异步通知与异步I/O