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

单例模式-如何保证全局唯一性?

以下是几种实现单例模式并保证全局唯一性的方法:

1. 饿汉式单例模式

class Singleton {
private:// 私有构造函数,防止外部创建对象Singleton() {}// 静态成员变量,存储单例对象static Singleton instance;
public:// 公有静态成员函数,用于获取单例对象static Singleton& getInstance() {return instance;}
};// 静态成员变量的初始化,在程序启动时创建单例对象
Singleton Singleton::instance;

解释

  • 构造函数私有化:将 Singleton 类的构造函数声明为 private,确保外部无法直接创建该类的对象。
  • 静态成员变量:使用 static Singleton instance 存储单例对象。
  • 静态成员函数:通过 static Singleton& getInstance() 提供获取单例对象的接口。
  • 全局初始化:在程序启动时,Singleton::instance 就会被创建,因为它是静态成员,且在类外进行了定义和初始化。这保证了在程序运行的任何时刻,getInstance() 都能返回同一个对象。

2. 懒汉式单例模式(线程不安全)

class Singleton {
private:// 私有构造函数,防止外部创建对象Singleton() {}// 静态成员变量,存储单例对象static Singleton* instance;
public:// 公有静态成员函数,用于获取单例对象static Singleton* getInstance() {if (instance == nullptr) {instance = new Singleton();}return instance;}
};// 静态成员变量的初始化,初始化为 nullptr
Singleton* Singleton::instance = nullptr;

解释

  • 构造函数私有化:与饿汉式相同,将构造函数设为 private
  • 静态指针成员变量:使用 static Singleton* instance 存储单例对象的指针,初始化为 nullptr
  • 静态成员函数getInstance() 函数在首次调用时检查 instance 是否为 nullptr,若为 nullptr 则创建对象,后续调用都将返回同一个对象。
  • 线程不安全:这种方式在多线程环境下存在问题,多个线程可能同时检查到 instance 为 nullptr,并同时创建对象,导致多个实例的出现。

3. 懒汉式单例模式(线程安全,使用互斥锁)

#include <mutex>class Singleton {
private:// 私有构造函数,防止外部创建对象Singleton() {}// 静态成员变量,存储单例对象static Singleton* instance;// 互斥锁,用于线程同步static std::mutex mtx;
public:// 公有静态成员函数,用于获取单例对象static Singleton* getInstance() {std::unique_lock<std::mutex> lock(mtx);if (instance == nullptr) {instance = new Singleton();}return instance;}
};// 静态成员变量的初始化,初始化为 nullptr
Singleton* Singleton::instance = nullptr;
// 静态成员变量的初始化,创建互斥锁
std::mutex Singleton::mtx;

解释

  • 构造函数私有化:确保外部无法直接创建对象。
  • 静态指针成员变量:存储单例对象的指针,初始化为 nullptr
  • 互斥锁:使用 std::mutex mtx 进行线程同步。
  • 静态成员函数:在 getInstance() 中,使用 std::unique_lock 获取互斥锁,确保同一时刻只有一个线程可以创建单例对象。
  • 性能问题:这种方式每次调用 getInstance() 都需要加锁,即使 instance 已经创建,会影响性能。

4. 懒汉式单例模式(线程安全,双重检查锁定)

#include <mutex>class Singleton {
private:// 私有构造函数,防止外部创建对象Singleton() {}// 静态成员变量,存储单例对象static Singleton* instance;// 互斥锁,用于线程同步static std::mutex mtx;
public:// 公有静态成员函数,用于获取单例对象static Singleton* getInstance() {if (instance == nullptr) {std::unique_lock<std::mutex> lock(mtx);if (instance == nullptr) {instance = new Singleton();}}return instance;}
};// 静态成员变量的初始化,初始化为 nullptr
Singleton* Singleton::instance = nullptr;
// 静态成员变量的初始化,创建互斥锁
std::mutex Singleton::mtx;

 解释

  • 构造函数私有化:防止外部创建对象。
  • 静态指针成员变量:存储单例对象指针,初始化为 nullptr
  • 互斥锁:用于线程同步。
  • 静态成员函数:使用双重检查锁定,第一次检查 instance 是否为 nullptr 是无锁的,若为 nullptr 则加锁再次检查并创建对象,避免了每次调用 getInstance() 都加锁的性能问题。
  • 潜在问题:在某些编译器和处理器架构下,由于指令重排,可能导致 instance 已经分配内存但未完成构造函数调用,导致其他线程访问到未完全初始化的对象。

5. 懒汉式单例模式(线程安全,C++11 及以上) 

#include <memory>
#include <mutex>class Singleton {
private:// 私有构造函数,防止外部创建对象Singleton() {}
public:// 公有静态成员函数,用于获取单例对象static Singleton& getInstance() {static std::once_flag flag;std::call_once(flag, []() {instance.reset(new Singleton());});return *instance.get();}
private:// 静态成员变量,存储单例对象static std::unique_ptr<Singleton> instance;
};// 静态成员变量的初始化,使用智能指针存储单例对象
std::unique_ptr<Singleton> Singleton::instance;

 解释

  • 构造函数私有化:防止外部创建对象。
  • 静态成员函数:使用 std::once_flag 和 std::call_once 保证单例对象只被创建一次。
  • 智能指针:使用 std::unique_ptr<Singleton> instance 存储单例对象,避免了内存管理问题。
  • C++11 特性std::call_once 是 C++11 引入的,确保 instance 只会被调用一次,且是线程安全的,解决了双重检查锁定的指令重排问题。

6、总结

通过以上几种方式,可以实现单例模式并保证全局唯一性。在实际应用中,推荐使用 C++11 及以上的 std::call_once 和 std::unique_ptr 实现,它提供了简洁、安全和高效的单例模式实现方式。


上述几种实现方式都旨在保证单例模式的全局唯一性,但各有优缺点,你可以根据实际需求和开发环境选择合适的实现方式。**代码解释**:
- **饿汉式单例模式**:- 优点:实现简单,在程序启动时就创建单例对象,保证了线程安全和全局唯一性。- 缺点:可能会造成资源浪费,因为单例对象在程序开始时就创建,无论是否使用。- **懒汉式单例模式(线程不安全)**:- 优点:单例对象在首次使用时创建,避免了资源浪费。- 缺点:在多线程环境下无法保证单例对象的唯一性,会出现多个实例。- **懒汉式单例模式(线程安全,使用互斥锁)**:- 优点:使用互斥锁保证了多线程环境下的单例对象唯一性。- 缺点:性能开销大,每次调用 `getInstance()` 都要加锁,即使单例对象已经创建。- **懒汉式单例模式(双重检查锁定)**:- 优点:在多线程环境下相对高效,避免了每次调用都加锁。- 缺点:存在指令重排的潜在风险,可能导致获取到未完全初始化的对象。- **懒汉式单例模式(线程安全,C++11及以上)**:- 优点:结合了 `std::call_once` 和 `std::unique_ptr`,既保证了线程安全,又避免了指令重排问题,是现代 C++ 推荐的实现方式。- 缺点:需要 C++11 及以上标准支持。

http://www.lryc.cn/news/518035.html

相关文章:

  • 设计模式学习笔记——结构型模式
  • WEB攻防-通用漏洞_文件上传_黑白盒审计流程
  • RabbitMQ基本介绍及简单上手
  • 服务器证书不受信任是什么问题?
  • spring mvc源码学习笔记之十
  • Ubuntu 下载安装 elasticsearch7.17.9
  • Qt笔记:网络编程Tcp
  • C++单例模式跨DLL调用问题梳理
  • oracle闪回版本查询
  • C#用winform窗口程序操作服务+不显示Form窗体,只显示右下角托盘图标+开机时自启动程序【附带项目地址】
  • UOS系统和windows系统wps文档显示差异问题解决
  • JS中函数基础知识之查漏补缺(写给小白的学习笔记)
  • 蓝桥杯训练
  • 前端学习DAY33(外边距的折叠)
  • asp.net core mvc的 ViewBag , ViewData , Module ,TempData
  • Linux驱动学习之第二个驱动程序(LED点亮关闭驱动程序-分层设计思想,使其能适应不同的板子-驱动程序模块为多个源文件怎么写Makefile)
  • 手写@EnableTransactionalManagement
  • 【Vue】:解决动态更新 <video> 标签 src 属性后视频未刷新的问题
  • 网络基础1 http1.0 1.1 http/2的演进史
  • Python 通过命令行在 unittest.TestCase 中运行单元测试
  • 源代码编译安装X11及相关库、vim,配置vim(2)
  • 设计模式 行为型 观察者模式(Observer Pattern)与 常见技术框架应用 解析
  • 【25考研】川大计算机复试情况,重点是啥?怎么准备?
  • C#调用Lua
  • LeetCode100之组合总和(39)--Java
  • NTN学习笔记之术语和缩写词解析
  • Yolo11改进:注意力改进|Block改进|ESSAformer,用于高光谱图像超分辨率的高效Transformer|即插即用
  • STM32 单片机 练习项目 LED灯闪烁LED流水灯蜂鸣器 未完待续
  • 使用PyTorch实现基于稀疏编码的生成对抗网络(GAN)在CIFAR-10数据集上的应用
  • 用matlab调用realterm一次性发送16进制数