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

C++:设计模式-单例模式

单例模式(Singleton Pattern)是一种设计模式,确保一个类只有一个实例,并且提供全局访问点。实现单例模式的关键是防止类被多次实例化,且能够保证实例的唯一性。常见的实现手法包括懒汉式饿汉式线程安全的懒汉式等。

1. 饿汉式(Eager Initialization)

饿汉式单例在程序启动时就创建实例,并且保证只有一个实例。适用于单例实例比较简单、没有资源消耗问题的情况。

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;

注意
Singleton Singleton::instance;应在类的实现文件(.cpp 文件)中定义。如果静态成员变量的定义(如 Singleton Singleton::instance;)也放在头文件中,由于头文件可能被多个源文件包含,就会导致多重定义错误。

优点

  • 简单直接,保证了类的实例唯一。
  • 实例在程序启动时就被创建,不会受到多线程的影响。

缺点

  • 无法延迟实例化。即使单例没有被使用,实例也会在程序启动时创建,可能会浪费资源。

在饿汉式单例中,实例在程序启动时就被创建,因此你无需显式地调用 getInstance() 来创建对象。它是静态的,并且一开始就存在。你只能通过 getInstance() 方法来访问该实例。

以下是如何在代码中调用饿汉式单例的示例:

完整代码示例
#include <iostream>
using namespace std;

class Singleton {
public:
    // 提供静态的访问方式
    static Singleton& getInstance() {
        return instance;  // 直接返回静态实例
    }

    void showMessage() {
        cout << "Singleton instance is working!" << endl;
    }

private:
    Singleton() { cout << "Singleton Constructor Called!" << endl; }  // 构造函数私有化
    ~Singleton() { cout << "Singleton Destructor Called!" << endl; }

    Singleton(const Singleton&) = delete;  // 禁止拷贝构造函数
    Singleton& operator=(const Singleton&) = delete;  // 禁止赋值

    static Singleton instance;  // 静态实例,程序启动时创建
};

// 静态实例必须定义在类外
Singleton Singleton::instance;

int main() {
    // 通过调用 getInstance 来获取单例实例
    Singleton& s1 = Singleton::getInstance();
    s1.showMessage();

    // 由于我们不能复制实例,以下代码会编译错误:
    // Singleton s2 = Singleton::getInstance();  // 编译错误,不能复制单例

    // 单例是全局唯一的,只能通过 getInstance() 获取
    return 0;
}
代码解析:
  1. 静态实例定义

    • static Singleton instance; 在类内部声明了一个静态的实例,该实例在程序启动时被创建。
    • 静态变量 instance 的生命周期从程序开始直到程序结束,所以它是全局唯一的。
  2. getInstance() 方法

    • static Singleton& getInstance() 提供了一个全局的访问点,用来获取唯一的实例。这个方法返回一个对静态成员 instance 的引用。
  3. main() 中的使用

    • main() 函数中,我们通过 Singleton::getInstance() 获取了单例实例,并调用了实例的方法 showMessage()
    • 单例实例的构造函数在第一次调用 getInstance() 时自动执行,但我们并没有显式地创建 Singleton 对象。
输出:
Singleton Constructor Called!
Singleton instance is working!
注意事项:
  • 全局唯一Singleton::getInstance() 返回的是同一个对象,所以每次调用 getInstance() 都会得到相同的实例。
  • 构造函数:构造函数只在第一次调用 getInstance() 时执行一次,因此 Singleton Constructor Called! 只会打印一次。
  • 禁止复制:拷贝构造函数和赋值运算符被删除,避免了实例的复制。编译时如果尝试复制单例实例(如 Singleton s2 = Singleton::getInstance();),会导致编译错误。
总结:

通过 Singleton::getInstance() 方法获取单例实例,这是调用饿汉式单例的标准方式。由于饿汉式实例在程序启动时就被创建,所以你不需要显式地进行实例化操作。


2. 懒汉式(Lazy Initialization)

懒汉式是在需要实例时才创建实例,这样可以延迟实例化,提高资源的使用效率。基本实现没有线程安全,多个线程同时访问时可能会出现问题。

class Singleton {
public:
    static Singleton* getInstance() {
        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;

优点

  • 实例化延迟,只有在第一次使用时才创建实例,避免了不必要的资源消耗。

缺点

  • 不是线程安全的。在多线程环境下,可能会创建多个实例。

3. 线程安全的懒汉式

为了确保线程安全,可以使用互斥锁(mutex)来同步访问,确保只有一个线程能够创建实例。

#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 Singleton* instance;
    static std::mutex mutex;  // 互斥锁
};

// 静态实例初始化为空指针
Singleton* Singleton::instance = nullptr;
std::mutex Singleton::mutex;

优点

  • 线程安全,保证在多线程环境下只有一个实例被创建。

缺点

  • 使用互斥锁增加了性能开销,尤其是频繁访问 getInstance() 时。

4. 双重检查锁(Double-Checked Locking)

双重检查锁定是懒汉式的优化版本,它减少了锁的使用频率。在第一次检查时不加锁,只有当实例为 nullptr 时才加锁,这样可以避免每次调用 getInstance() 时都进行加锁操作。

#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 Singleton* instance;
    static std::mutex mutex;  // 互斥锁
};

// 静态实例初始化为空指针
Singleton* Singleton::instance = nullptr;
std::mutex Singleton::mutex;

优点

  • 提供了线程安全的懒汉式实现,且避免了每次都加锁的性能开销。

缺点

  • 代码相对复杂,需要小心实现。对 instance 的访问需要特别注意线程间的同步。

5. 静态局部变量(最推荐的实现方式)

使用静态局部变量来实现单例模式,这种方式是线程安全的,并且实现简单。C++11标准及以上可以确保静态局部变量的初始化是线程安全的。

class Singleton {
public:
    static Singleton& getInstance() {
        static Singleton instance;  // 静态局部变量
        return instance;
    }

private:
    Singleton() {}  // 构造函数私有化
    ~Singleton() {}

    Singleton(const Singleton&) = delete;  // 禁止拷贝构造函数
    Singleton& operator=(const Singleton&) = delete;  // 禁止赋值
};

优点

  • 简洁、线程安全。
  • 在程序启动时不会立即创建实例,而是等到第一次使用时创建。
  • 不需要显式加锁。

缺点

  • 只适用于需要在第一次使用时实例化的情况。

总结

  • 饿汉式:简单,适合不需要延迟初始化的场景,程序启动时就创建实例。
  • 懒汉式:适合需要延迟初始化的场景,但需要考虑线程安全。
  • 线程安全懒汉式:通过加锁保证线程安全,但可能带来性能开销。
  • 双重检查锁:线程安全,减少了加锁的频率,但实现复杂。
  • 静态局部变量:线程安全,简洁,现代 C++ 中推荐的单例实现方式。

在现代 C++ 中,静态局部变量是实现单例模式的首选方法,既保证了线程安全,又没有性能开销,是最优选择。


http://www.kler.cn/a/405779.html

相关文章:

  • 人工智能(AI)与机器学习(ML)基础知识
  • 区块链应用到银行的优势
  • kafka中是如何快速定位到一个offset的
  • 贪心算法 -- 递增子序列
  • Kafka Offset 自动提交和手动提交 - 漏消费与重复消费
  • aws凭证(一)凭证存储
  • 如何优化Sql的查询性能?
  • 实景三维赋能国土空间智慧治理
  • Python爬虫中的IP封禁问题及其解决方案
  • Chrome 浏览器 131 版本新特性
  • WebLogic 版本升级的注意事项与流程
  • STM32 RAM在Memory Map中被分为3个区域
  • Vscode/Code-server无网环境安装通义灵码
  • Jenkins修改LOGO
  • 【大数据测试Flume:从 0-1详细教程】
  • Chrome 浏览器 131 版本开发者工具(DevTools)更新内容
  • 基于深度学习CNN算法的花卉分类识别系统01--带数据集-pyqt5UI界面-全套源码
  • 使用Electron将vue2项目打包为桌面exe安装包
  • 大数据基于Spring Boot的化妆品推荐系统的设计与实现
  • 深入探究蓝牙节能技术:SNIFF与HOLD模式
  • Spring Boot 的 WebClient 实践教程
  • DNS解析记录怎么查询?
  • 一个小的可编辑表格问题引起的思考
  • Java技术复习提升 10异常
  • 小红书运营教程10(小红书笔记制作01)
  • 安卓手机5G网络频繁掉4G 问题解决 手机5G网络优化方案