C++单例模式

局部静态变量方式

cpp 复制代码
//通过静态成员变量实现单例
//懒汉式
class Single2
{
private:
    Single2()
    {
    }
    Single2(const Single2 &) = delete;
    Single2 &operator=(const Single2 &) = delete;
public:
    static Single2 &GetInst()
    {
        static Single2 single;
        return single;
    }
};

一个经典的懒汉式单例模式实现,通过静态成员变量来确保 Single2 类只有一个实例。这个实现是正确的,但有几点需要注意:

  1. 线程安全性 :在C++11及更高版本中,局部静态变量(如 GetInst 方法中的 static Single2 single)的初始化是线程安全的。因此,这个实现是线程安全的,不会在多线程环境中引发问题。

  2. 构造函数和赋值运算符的删除 :你通过 delete 删除了拷贝构造函数和赋值运算符,这可以防止在外部创建类的拷贝或通过赋值创建多个实例,这是单例模式的正确做法。

  3. 懒汉式单例的使用 :懒汉式单例在首次调用 GetInst 时才会创建实例,这样可以避免程序启动时不必要的内存消耗。

在C++11以下的版本中,使用你提供的懒汉式单例实现可能会遇到线程安全问题。具体来说,问题出现在静态局部变量 single 的初始化上。

线程安全问题

在C++11之前,局部静态变量的初始化不是线程安全的。如果多个线程同时首次调用 GetInst 方法,有可能会发生竞态条件(race condition),导致创建出多个 Single2 实例,破坏了单例模式的初衷。

解决方法

如果需要在C++11以下的版本中确保线程安全,可以考虑以下几种方法:

  1. 双重检查锁定(Double-Checked Locking)
cpp 复制代码
class Single2
{
private:
    static Single2* instance;
    static std::mutex mtx;
    Single2() {}
    Single2(const Single2 &) = delete;
    Single2 &operator=(const Single2 &) = delete;

public:
    static Single2* GetInst()
    {
        if (instance == nullptr) // 第一次检查
        {
            std::lock_guard<std::mutex> lock(mtx); // 加锁
            if (instance == nullptr) // 第二次检查
            {
                instance = new Single2();
            }
        }
        return instance;
    }
};

Single2* Single2::instance = nullptr;
std::mutex Single2::mtx;
  1. 静态成员变量初始化(线程不安全版本)

如果你确定单例只会在单线程环境下使用,可以使用以下更简单的方式:

cpp 复制代码
class Single2
{
private:
    static Single2* instance;
    Single2() {}
    Single2(const Single2 &) = delete;
    Single2 &operator=(const Single2 &) = delete;

public:
    static Single2* GetInst()
    {
        if (instance == nullptr)
        {
            instance = new Single2();
        }
        return instance;
    }
};

Single2* Single2::instance = nullptr;

这种实现不适用于多线程环境。

C++11 引入了一些新的特性和改进,其中之一就是保证了局部静态变量的线程安全性。这意味着在 C++11 及更高版本中,局部静态变量的初始化是由编译器自动处理的,并且保证了这种初始化过程在多线程环境下是安全的。

C++11 之前的问题

在 C++11 之前,当多个线程同时首次访问一个局部静态变量时,可能会出现竞态条件(Race Condition),导致多个线程同时进入变量的初始化过程。由于这个初始化过程没有得到线程的同步保护,可能会导致初始化多次,进而破坏单例模式的唯一性要求。

C++11 的改进

C++11 标准引入了以下保证:

  • 局部静态变量的初始化是线程安全的:C++11 规定,如果多个线程试图同时初始化同一个局部静态变量,只有一个线程会成功初始化它,而其他线程会被阻塞,直到初始化完成。这就确保了静态局部变量在多线程环境下也只会被初始化一次。

  • 懒汉式单例的线程安全性:由于局部静态变量的初始化已经是线程安全的,在 C++11 及更高版本中,可以安全地使用局部静态变量来实现懒汉式单例模式,无需额外的锁机制来防止竞态条件。

C++11 线程安全的实现机制

在底层,C++11 通过引入新的机制,确保了局部静态变量初始化的线程安全。这些机制包括:

  1. 线程同步机制:C++11 在初始化局部静态变量时,使用了编译器提供的同步机制(如锁、屏障等)来确保在多线程环境下,只有一个线程可以执行初始化代码。

  2. 按需初始化:C++11 确保局部静态变量在首次使用时才会被初始化,这种方式不仅节省资源,还保证了多线程环境中的唯一性。

相关推荐
k要开心5 分钟前
C++概念以及基础框架语法
开发语言·c++
LUCIAZZZ29 分钟前
HikariCP数据库连接池原理解析
java·jvm·数据库·spring·springboot·线程池·连接池
sky_ph1 小时前
JAVA-GC浅析(二)G1(Garbage First)回收器
java·后端
开发者工具分享1 小时前
如何应对敏捷转型中的团队阻力
开发语言
gregmankiw1 小时前
C#调用Rust动态链接库DLL的案例
开发语言·rust·c#
IDRSolutions_CN1 小时前
PDF 转 HTML5 —— HTML5 填充图形不支持 Even-Odd 奇偶规则?(第二部分)
java·经验分享·pdf·软件工程·团队开发
hello早上好1 小时前
Spring不同类型的ApplicationContext的创建方式
java·后端·架构
roman_日积跬步-终至千里1 小时前
【Go语言基础【20】】Go的包与工程
开发语言·后端·golang
秦少游在淮海2 小时前
C++ - string 的使用 #auto #范围for #访问及遍历操作 #容量操作 #修改操作 #其他操作 #非成员函数
开发语言·c++·stl·string·范围for·auto·string 的使用
const5442 小时前
cpp自学 day2(—>运算符)
开发语言·c++