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

news/2025/1/12 13:32:46/

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

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.ppmy.cn/news/1562517.html

相关文章

【git命令】--- git经典常用操作命令大全

在编程的艺术世界里&#xff0c;代码和灵感需要寻找到最佳的交融点&#xff0c;才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里&#xff0c;我们将共同追寻这种完美结合&#xff0c;为未来的世界留下属于我们的独特印记。 【git命令】--- git经典常用操作命令大全…

Redis 的大 Key 对持久化有什么影响

Redis 的持久化方式有两种&#xff1a;AOF 日志和 RDB 快照。 所以接下来&#xff0c;针对这两种持久化方式具体分析分析。 大 Key 对 AOF 日志的影响 先说说 AOF 日志三种写回磁盘的策略 Redis 提供了 3 种 AOF 日志写回硬盘的策略&#xff0c;分别是&#xff1a; Always&am…

golang观察者设计模式

观察者模式简介 观察者模式&#xff08;Observer Pattern&#xff09;是一种行为型设计模式&#xff0c;它定义了一种一对多的依赖关系&#xff0c;让多个观察者对象同时监听某一个主题对象。当主题对象的状态发生变化时&#xff0c;所有依赖于它的观察者对象都会得到通知并自…

73.矩阵置零 python

矩阵置零 题目题目描述示例 1&#xff1a;示例 2&#xff1a;提示&#xff1a; 题解思路分析Python 实现代码代码解释提交结果 题目 题目描述 给定一个 m x n 的矩阵&#xff0c;如果一个元素为 0 &#xff0c;则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。 示例…

网络编程的进程查看连接描述符信息等

一.查看当前进程的socket对应的fd信息 1. lsof lsof&#xff08;List Open Files&#xff09;命令可以列出系统中所有打开的文件的信息&#xff0c;包括 socket。 用法 要查看特定进程的 socket 信息&#xff0c;可以使用以下命令&#xff1a; lsof -p <pid> | grep…

如何构建多层决策树

构建一颗多层的决策树时&#xff0c;通过递归选择最佳划分特征&#xff08;依据 信息增益 或 基尼系数&#xff09;对数据集进行划分&#xff0c;直到满足停止条件&#xff08;例如叶节点纯度达到要求或树的深度限制&#xff09;。以下是基于 信息增益 和 基尼系数 的递推公式和…

12工具篇(D2_Commons-lang3)

目录 一、基本介绍 二、commons-lang3库的引用 三、常用工具类 1. StringUtils 2. RandomStringUtils 3. NumberUtils 4. ArrayUtils 5. DateUtils 6. StringUtile 7. BeanUtils 7.1 基本介绍 7.2 JavaBean 7.3 两个概念 7.4 常用操作 7.5 应用 1. JavaBean 的属…

ISP架构方案

外置 ISP 架构 外置 ISP 架构是指在 AP 外部单独布置 ISP 芯片用于图像信号处理。外置 ISP 的架构图一般如下所示&#xff1a; 外置 ISP 架构的优点主要有&#xff1a; 能够提供更优秀的图像质量&#xff1a;在激烈的市场竞争下&#xff0c;能够存活到现在的外置 ISP 生产厂商在…