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

【无标题】C++单例模式详解

文章目录

  • 一、单例模式的核心要求
  • 二、常见实现方式
    • 1. 懒汉式(Lazy Initialization)
    • 2. 饿汉式(Eager Initialization)
    • 3. 线程安全的懒汉式(双检锁模式,Double-Checked Locking)
    • 4. Meyers’ Singleton(C++11 最佳实践)
  • 三、线程安全与内存管理
    • 1. 线程安全分析
    • 2. 内存泄漏问题
    • 3. 自动释放机制
  • 四、单例模式的优缺点
    • 优点
    • 缺点
  • 五、高级话题
    • 1. 模板单例(通用实现)
    • 2. 多线程下的性能优化
    • 3. 单例与序列化
  • 六、使用场景
  • 七、注意事项
  • 总结

概述 C++ 单例模式(SingletonPattern)是一种创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。该模式常用于日志记录器、配置管理器、数据库连接池等场景。以下从实现方式、线程安全、内存管理等方面进行详细解析。

一、单例模式的核心要求

私有构造函数:防止外部通过 new 创建实例。
静态实例变量:存储类的唯一实例。
全局访问方法:提供获取实例的静态接口。
禁止拷贝和赋值:防止通过复制或赋值创建新实例。

二、常见实现方式

1. 懒汉式(Lazy Initialization)

特点:延迟实例化,首次使用时创建。
问题:线程不安全,多线程环境下可能创建多个实例。

class LazySingleton {
private:static LazySingleton* instance;  // 静态实例指针LazySingleton() = default;       // 私有构造函数~LazySingleton() = default;LazySingleton(const LazySingleton&) = delete;             // 禁用拷贝构造LazySingleton& operator=(const LazySingleton&) = delete;  // 禁用赋值运算符public:static LazySingleton* getInstance() {if (instance == nullptr) {  // 首次调用时创建实例instance = new LazySingleton();}return instance;}
};// 静态成员变量需要在类外初始化
LazySingleton* LazySingleton::instance = nullptr;

2. 饿汉式(Eager Initialization)

特点:程序启动时立即创建实例,线程安全。
缺点:无论是否使用,实例都会被创建,可能浪费资源。

class EagerSingleton {
private:static EagerSingleton instance;  // 静态实例对象EagerSingleton() = default;~EagerSingleton() = default;EagerSingleton(const EagerSingleton&) = delete;EagerSingleton& operator=(const EagerSingleton&) = delete;public:static EagerSingleton& getInstance() {return instance;  // 直接返回已创建的实例}
};// 类外初始化,程序启动时即创建实例
EagerSingleton EagerSingleton::instance;

3. 线程安全的懒汉式(双检锁模式,Double-Checked Locking)

特点:通过双重检查和互斥锁保证线程安全,减少锁竞争开销。

#include <mutex>
class ThreadSafeSingleton {
private:static ThreadSafeSingleton* instance;static std::mutex mutex_;ThreadSafeSingleton() = default;~ThreadSafeSingleton() = default;ThreadSafeSingleton(const ThreadSafeSingleton&) = delete;ThreadSafeSingleton& operator=(const ThreadSafeSingleton&) = delete;public:static ThreadSafeSingleton* getInstance() {if (instance == nullptr) {  // 第一次检查,无锁std::lock_guard<std::mutex> lock(mutex_);  // 加锁if (instance == nullptr) {  // 第二次检查,确保只创建一次instance = new ThreadSafeSingleton();}}return instance;}
};ThreadSafeSingleton* ThreadSafeSingleton::instance = nullptr;
std::mutex ThreadSafeSingleton::mutex_;

4. Meyers’ Singleton(C++11 最佳实践)

特点:使用局部静态变量,线程安全(C++11 起),自动处理内存释放。

class MeyersSingleton {
private:MeyersSingleton() = default;~MeyersSingleton() = default;MeyersSingleton(const MeyersSingleton&) = delete;MeyersSingleton& operator=(const MeyersSingleton&) = delete;public:static MeyersSingleton& getInstance() {static MeyersSingleton instance;  // 局部静态变量,线程安全return instance;}
};

三、线程安全与内存管理

1. 线程安全分析

懒汉式:非线程安全,需手动加锁。
饿汉式:线程安全,由静态初始化保证。
双检锁模式:线程安全,但需注意内存模型(C++11 后通过 std::atomic 修复)。
Meyers’ Singleton:线程安全,C++11 标准保证静态变量初始化的原子性。

2. 内存泄漏问题

普通懒汉式:使用 new 创建实例,需手动 delete 或通过智能指针管理。
智能指针方案:

class SmartSingleton {
private:static std::unique_ptr<SmartSingleton> instance;SmartSingleton() = default;~SmartSingleton() = default;SmartSingleton(const SmartSingleton&) = delete;SmartSingleton& operator=(const SmartSingleton&) = delete;public:static SmartSingleton& getInstance() {if (instance == nullptr) {instance = std::make_unique<SmartSingleton>();}return *instance;}
};std::unique_ptr<SmartSingleton> SmartSingleton::instance = nullptr;

3. 自动释放机制

静态对象:如 Meyers’ Singleton,程序结束时自动析构。
atexit 注册:在单例类中注册析构函数:

static void destroyInstance() {delete instance;instance = nullptr;
}// 在 getInstance() 中首次创建实例时注册
std::atexit(destroyInstance);

四、单例模式的优缺点

优点

全局唯一实例:确保系统中只有一个实例,便于资源控制。
全局访问点:无需传递对象引用,直接通过静态方法访问。
延迟初始化(懒汉式):按需创建,节省资源。

缺点

违反单一职责原则:类既要管理自身生命周期,又要提供业务功能。
难以测试:单例状态可能影响测试的独立性,Mock 实现困难。
多线程风险:若实现不当,易导致线程安全问题。
潜在内存泄漏:手动管理实例生命周期时需谨慎。

五、高级话题

1. 模板单例(通用实现)

template <typename T>
class Singleton {
protected:Singleton() = default;~Singleton() = default;Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;public:static T& getInstance() {static T instance;return instance;}
};// 使用示例
class MyClass : public Singleton<MyClass> {friend class Singleton<MyClass>;  // 允许基类访问构造函数
private:MyClass() = default;  // 私有构造函数
public:void doSomething() { /* ... */ }
};

2. 多线程下的性能优化

减少锁竞争:双检锁模式仅在首次创建时加锁。
无锁实现:C++11 后使用 std::atomic 和内存屏障:

std::atomic<Singleton*> Singleton::instance{nullptr};
std::mutex Singleton::mutex_;Singleton* Singleton::getInstance() {Singleton* tmp = instance.load(std::memory_order_relaxed);std::atomic_thread_fence(std::memory_order_acquire);if (tmp == nullptr) {std::lock_guard<std::mutex> lock(mutex_);tmp = instance.load(std::memory_order_relaxed);if (tmp == nullptr) {tmp = new Singleton;std::atomic_thread_fence(std::memory_order_release);instance.store(tmp, std::memory_order_relaxed);}}return tmp;
}

3. 单例与序列化

若单例类需支持序列化,需确保反序列化时不创建新实例:

class SerializableSingleton {
private:friend class boost::serialization::access;template<class Archive>void serialize(Archive & ar, const unsigned int version) {// 序列化逻辑}protected:SerializableSingleton() = default;private:// 防止反序列化时创建新实例friend class boost::serialization::singleton;static SerializableSingleton& get_instance();
};

六、使用场景

资源管理器:如文件系统、数据库连接池。
配置管理:全局配置信息的读写。
日志记录器:统一记录系统日志,避免多实例冲突。
GUI 应用:主窗口、对话框等通常设计为单例。

七、注意事项

避免滥用:单例易导致全局状态,增加代码耦合度。
多进程/分布式系统:单例仅在单个进程内有效,跨进程需使用其他机制(如共享内存、分布式锁)。
继承与多态:单例类的子类可能破坏单例特性,需谨慎设计。

总结

C++ 单例模式的最佳实践推荐使用 Meyers’ Singleton(C++11 及以后),其简洁、线程安全且自动管理内存。对于旧版 C++ 或需要更复杂控制的场景,可采用双检锁模式并结合智能指针。无论选择哪种实现,都需注意线程安全、内存管理和代码可测试性。

http://www.xdnf.cn/news/9195.html

相关文章:

  • 二次封装 Vuex for Uniapp 微信小程序开发
  • linux如何查看网络设备类型
  • 学者观察 | Web3.0的技术革新与挑战——北京理工大学教授沈蒙
  • 机器学习中的关键术语及其含义
  • 打造自己的开源组件:如何将 Starter 发布到 Maven Central?
  • 人工智能100问☞第34问:什么是语音识别与合成?
  • xilinx 7系列底层可配置逻辑块CLB资源简介
  • js 实现多并发任务处理
  • AI时代的弯道超车之第二十一章:AI会颠覆哪些行业?
  • 什么是MCP技术,跟http技术有什么区别
  • Excel 统计某个字符串在指定区域出现的次数
  • 低空经济管理系统设计方案
  • Spring Boot 3.4.6 中文文档上线
  • 深入理解 JDK、JRE 和 JVM 的区别
  • CellularPro 1.8.6.1 | 提升网络速度,抢到更多基站的速度
  • Netty创新架构突破链接数瓶颈技术,如何应用于新能源汽车智慧充电桩?
  • Redis 容器启动失败Fatal error loading the DB, check server logs. Exiting.的解决方法
  • 使用 ssld 提取CMS 签名并重签名
  • 在PyTorch中,有了y = x + y,为什么还需要y += x,有什么好处呢?
  • 九级融智台阶的要素协同跃迁框架
  • 6个月Python学习计划 Day 6 - 综合实战:学生信息管理系统
  • ai写歌平台:AnKo开启音乐创作的智能时代!
  • java类加载器
  • 树莓派超全系列教程文档--(50)如何查找树莓派的IP地址
  • 计算机组成与体系结构:硬盘驱动器(Hard Disk Drives)
  • OpenGL Chan视频学习-9 Index Buffers inOpenGL
  • STM32F407VET6学习笔记6:定时器TIM2的配置使用
  • MPLS实验复现
  • 70页精品PPT | 休闲食品行业数据分析平台建设方案快消BI大数据解决方案BI方案
  • [ Qt ] | 常用控件(三):