C++单例模式

在C++中,单例模式是一种确保一个类只有一个实例,并提供一个全局访问点来访问这个实例的设计模式。

单例模式中的饿汉式(Eager Initialization)和懒汉式(Lazy Initialization)是两种不同的实例化策略,它们在实例的创建时机、性能和线程安全性方面有所区别。

以下是一个单例模式的典型实现:

饿汉式(Eager Initialization)

  1. 实例创建时机

    • 在程序启动时或类加载时立即创建单例实例。
  2. 性能

    • 由于实例在程序开始时就已经创建,因此第一次调用 getInstance 方法时不需要进行任何同步操作,速度较快。
  3. 线程安全性

    • 饿汉式是线程安全的,因为实例在类加载时就已经创建,JVM(Java虚拟机)保证了类加载过程的线程安全性。
  4. 资源使用

    • 如果单例实例很大或者初始化过程很耗时,并且程序可能永远不使用这个实例,那么饿汉式可能会浪费资源。
  5. 实现

    • 通常通过静态常量实现。

在这种方法中,单例实例在程序开始时就被创建。

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

private:
    Singleton() {}
    static Singleton instance; // 静态常量
};

Singleton Singleton::instance; // 在程序启动时创建实例

懒汉式(Lazy Initialization)

  1. 实例创建时机

    • 在第一次调用 getInstance 方法时创建单例实例。
  2. 性能

    • 第一次调用 getInstance 时需要创建实例,可能会有一定的延迟,但如果实例很大或者初始化很耗时,那么这种方式可以延迟资源的使用,提高性能。
  3. 线程安全性

    • 基本的懒汉式实现不是线程安全的,因为在多线程环境中可能会有多个线程同时进入 if 判断并创建多个实例。
    • 为了保证线程安全,通常需要添加同步机制(如互斥锁),但这会降低性能。
  4. 资源使用

    • 懒汉式只在需要时创建实例,更加节省资源。
  5. 实现

    • 通常通过静态成员变量和同步代码块实现。
cpp 复制代码
class Singleton {
public:
    static Singleton* getInstance() {
        if (instance == nullptr) {
            instance = new Singleton();
        }
        return instance;
    }

private:
    Singleton() {}
    static Singleton* instance; // 静态成员变量
};

Singleton* Singleton::instance = nullptr; // 初始时为空

为了使懒汉式线程安全,可以添加互斥锁:

cpp 复制代码
#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() {}
    static Singleton* instance; // 静态成员变量
    static std::mutex mutex_;  // 互斥锁
};

Singleton* Singleton::instance = nullptr;
std::mutex Singleton::mutex_;

懒汉式使用示例

cpp 复制代码
Student* Student::instance_ = NULL;
Student Student::staticInstance;

class Student: public people
{
public:
    static Student * instance(){return instance_;};

private:

    Student ();
    virtual ~Student ();

    static Student * instance_;
    static Student staticInstance;

    Student (const Student &);
    Student & operator = (const Student &);
};
Student ::Student ():
{
    instance_ = &staticInstance;
}
相关推荐
じ☆ve 清风°1 小时前
JavaScript 原型与原型链:深入理解 __proto__ 和 prototype 的由来与关系
开发语言·javascript·原型模式
_r0bin_6 小时前
前端面试准备-7
开发语言·前端·javascript·fetch·跨域·class
zhang98800006 小时前
JavaScript 核心原理深度解析-不停留于表面的VUE等的使用!
开发语言·javascript·vue.js
Fanxt_Ja7 小时前
【JVM】三色标记法原理
java·开发语言·jvm·算法
蓝婷儿7 小时前
6个月Python学习计划 Day 15 - 函数式编程、高阶函数、生成器/迭代器
开发语言·python·学习
love530love8 小时前
【笔记】在 MSYS2(MINGW64)中正确安装 Rust
运维·开发语言·人工智能·windows·笔记·python·rust
南郁8 小时前
007-nlohmann/json 项目应用-C++开源库108杰
c++·开源·json·nlohmann·现代c++·d2school·108杰
slandarer8 小时前
MATLAB | 绘图复刻(十九)| 轻松拿捏 Nature Communications 绘图
开发语言·matlab
狐凄8 小时前
Python实例题:Python计算二元二次方程组
开发语言·python
roman_日积跬步-终至千里9 小时前
【Go语言基础【3】】变量、常量、值类型与引用类型
开发语言·算法·golang