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

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

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

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

基本思路

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

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

立即学习“C++免费学习笔记(深入)”;

#include 

class Singleton {public:// 获取唯一实例static Singleton* getInstance() {// 第一次检查:避免不必要的加锁if (instance == nullptr) {std::lockguard 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起线程安全):

立即学习“C++免费学习笔记(深入)”;

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

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

总结

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

基本上就这些。

以上就是C++单例模式实现 线程安全双重检查锁的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1471874.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 19:10:05
下一篇 2025年12月18日 19:10:20

相关推荐

发表回复

登录后才能评论
关注微信