C++单例模式在多线程环境下如何保证线程安全?

在多线程环境下保证单例模式的线程安全,可以采用以下几种方法:

饿汉式:在程序启动时就创建单例实例,这种方式简单且线程安全,因为实例是在程序启动时创建的,所以不会有线程安全问题。但是它的缺点是不管是否使用实例都会占用资源。

cpp 复制代码
class Singleton {
public:
    static Singleton& getInstance() {
        return instance;
    }

private:
    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    static Singleton instance;
};

// 在类外初始化静态成员变量
Singleton Singleton::instance;

懒汉式 :在第一次使用时才创建实例,这种方式需要额外的同步机制来保证线程安全。可以通过加锁(互斥锁)来实现,例如使用std::mutexstd::lock_guard来确保线程安全。

cpp 复制代码
#include <mutex>

class Singleton {
public:
    static Singleton& getInstance() {
        static std::mutex mutex;
        if (instance == nullptr) {
            std::lock_guard<std::mutex> lock(mutex);
            if (instance == nullptr) {
                instance = new Singleton();
            }
        }
        return *instance;
    }

private:
    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    static Singleton* instance;
};

// 在类外初始化静态成员指针
Singleton* Singleton::instance = nullptr;

双重检查锁定(Double-Checked Locking) :在获取实例的方法中,先检查实例是否已经创建,如果未创建则加锁并再次检查,这样可以减少锁的开销。但是这种方式需要配合std::atomic来保证内存的可见性和防止指令重排。

cpp 复制代码
#include <atomic>

class Singleton {
public:
    static Singleton& getInstance() {
        if (instance == nullptr) {
            std::lock_guard<std::mutex> lock(mutex);
            if (instance == nullptr) {
                instance = new Singleton();
            }
        }
        return *instance;
    }

private:
    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    static std::atomic<Singleton*> instance;
    static std::mutex mutex;
};

// 在类外初始化静态成员变量
std::atomic<Singleton*> Singleton::instance(nullptr);
std::mutex Singleton::mutex;

局部静态变量:在C++11及以后的版本中,局部静态变量的初始化是线程安全的。因此,可以在一个静态方法中创建一个局部静态变量来实现单例,这种方式既简单又线程安全。

cpp 复制代码
class Singleton {
public:
    static Singleton& getInstance() {
        static Singleton instance;
        return instance;
    }

private:
    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;
};

使用std::call_once:这是一种线程安全的方式来保证代码只执行一次,通常用于懒汉式的单例模式实现中。

cpp 复制代码
#include <mutex>

class Singleton {
public:
    static Singleton& getInstance() {
        static std::once_flag onceFlag;
        std::call_once(onceFlag, &Singleton::initInstance);
        return *_instance;
    }

private:
    static void initInstance() {
        _instance = new Singleton();
    }

    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    static Singleton* _instance;
    static std::once_flag onceFlag;
};

// 在类外初始化静态成员变量和once_flag
Singleton* Singleton::_instance = nullptr;
std::once_flag Singleton::onceFlag;

使用std::atomicstd::mutex:通过原子操作来确保只有一个线程可以执行实例创建的代码块。

cpp 复制代码
#include <atomic>
#include <mutex>

class Singleton {
public:
    static Singleton& getInstance() {
        if (instance == nullptr) {
            std::lock_guard<std::mutex> lock(mutex);
            if (instance == nullptr) {
                instance = new Singleton();
            }
        }
        return *instance;
    }

private:
    Singleton() {}
    ~Singleton() {}
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    static std::atomic<Singleton*> instance;
    static std::mutex mutex;
};

// 在类外初始化静态成员变量
std::atomic<Singleton*> Singleton::instance(nullptr);
std::mutex Singleton::mutex;
相关推荐
cookies_s_s30 分钟前
Linux--进程(进程虚拟地址空间、页表、进程控制、实现简易shell)
linux·运维·服务器·数据结构·c++·算法·哈希算法
不想编程小谭1 小时前
力扣LeetCode: 2506 统计相似字符串对的数目
c++·算法·leetcode
曼巴UE52 小时前
UE5.3 C++ TArray系列(一)
开发语言·c++·ue5
阿巴~阿巴~3 小时前
多源 BFS 算法详解:从原理到实现,高效解决多源最短路问题
开发语言·数据结构·c++·算法·宽度优先
CoderCodingNo4 小时前
【GESP】C++二级真题 luogu-b3924, [GESP202312 二级] 小杨的H字矩阵
java·c++·矩阵
刃神太酷啦4 小时前
堆和priority_queue
数据结构·c++·蓝桥杯c++组
Heris994 小时前
2.22 c++练习【operator运算符重载、封装消息队列、封装信号灯集】
开发语言·c++
----云烟----4 小时前
C/C++ 中 volatile 关键字详解
c语言·开发语言·c++
ChoSeitaku6 小时前
12.重复内容去重|添加日志|部署服务到Linux上(C++)
linux·c++·windows
挣扎与觉醒中的技术人6 小时前
网络安全入门持续学习与进阶路径(一)
网络·c++·学习·程序人生·安全·web安全