• 常用
  • 百度
  • google
  • 站内搜索

科技

C++单例模式实现,线程安全的双重检查锁机制

  • 更新日期:2025-11-29
  • 查看次数:3764

双重检查锁定通过两次检查和加锁确保线程安全且提升性能,C++11后推荐使用局部静态变量实现更安全简洁的单例模式。

C++单例模式实现 线程安全双重检查锁

在C++中实现线程安全的单例模式,双重检查锁定(Double-Checked Locking Pattern, DCLP)是一种常见且高效的方案。它既能保证性能(避免每次调用都加锁),又能确保多线程环境下只创建一个实例。

基本思路

双重检查锁定的核心是在获取实例时进行两次检查:第一次检查是否需要加锁,第二次在锁内检查是否真正需要创建对象,从而减少锁竞争。

以下是使用C++11及以上标准实现的线程安全单例模式(基于DCLP):

#include <mutex>

class Singleton { public: // 获取唯一实例 static Singleton* getInstance() { // 第一次检查:避免不必要的加锁 if (instance == nullptr) { std::lockguard<std::mutex> lock(mutex); // 第二次检查:确保只有一个线程创建实例 if (instance == nullptr) { instance = new Singleton(); } } return instance; }

// 删除拷贝构造和赋值操作
Singleton(const Singleton&) = delete;
Singleton& operator=(const Singleton&) = delete;

private: Singleton() = default; // 私有构造函数 ~Singleton() = default; // 可选析构

static Singleton* instance;
static std::mutex mutex_;

};

// 静态成员定义 Singleton* Singleton::instance = nullptr; std::mutex Singleton::mutex_;

为什么需要双重检查?

如果只做一次加锁检查,每次调用 getInstance() 都会进入临界区,影响性能。双重检查通过先判断指针是否为空,仅在必要时才加锁,提升了并发效率。

关键点:

  • 第一次检查无锁,快速返回已创建的实例
  • 加锁后再次检查,防止多个线程同时创建对象
  • C++11起,nullptr 和内存模型支持使得DCLP在正确实现下是安全的

注意内存释放问题

上面的实现使用 new 创建对象,但没有释放。若需自动释放,可结合智能指针或静态局部变量方式。

更现代的替代方案:使用局部静态变量(C++11起线程安全):

static Singleton& getInstance() {
    static Singleton instance; // 局部静态变量,线程安全且自动析构
    return instance;
}

这种方式更简洁、安全,编译器保证初始化的线程安全性,无需手动加锁。除非有特殊需求(如延迟初始化控制),推荐使用此方法。

总结

双重检查锁定可用于实现高性能的线程安全单例,但在C++11以后,优先考虑局部静态变量方式。若必须手动管理,注意使用互斥量和正确的空指针检查顺序。

基本上就这些。

本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。

imtoken下载 im钱包 imtoken imtoken 快连官网 imtoken imtoken imtoken imtoken imtoken wallet imtoken imtoken官网 imtoken钱包 imtoken下载 imtoken官网 imtoken钱包 imtoken安卓下载 imtoken下载 imtoken官方下载 imtoken官网 imtoken安卓下载 imtoken下载 imtoken下载 imtoken imtoken imtoken imtoken imtoken imtoken imtoken imtoken imtoken bitget wallet telegram下载 quickq VPN trust wallet v2rayn imtoken