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

单例模式的介绍

单例模式(Singleton)是一种创建型设计模式,它确保一个类只有一个实例,并提供全局访问点。其核心思想是通过限制类的实例化次数,防止多个实例同时存在,从而避免了多线程竞争和资源浪费,提高了代码的可维护性和可扩展性。

单例模式通常适用于以下场景:

  1. 系统中某个类只能存在一个实例,例如系统配置、日志管理器等。
  2. 需要频繁访问同一个对象或资源,但又不希望每次都去创建新的对象。
  3. 为了确保数据的一致性和完整性,需要限制全局变量的访问权限。

实现单例模式的方法有很多种,常见的包括懒汉式、饿汉式、双重检查锁等。具体实现方式可以根据实际情况选择,但需要注意线程安全性、效率等问题。

在 C++ 中,可以使用静态成员变量或者静态局部变量来实现单例模式。静态成员变量在程序启动时就会被初始化,因此是线程安全的,但需要手动释放资源。静态局部变量则是在首次调用时初始化,虽然不需要手动释放资源,但需要考虑线程安全性和可重入性等问题。

总之,单例模式是一种简单而又实用的设计模式,可以在很多场景下提高代码的可维护性和可扩展性。在使用单例模式时,需要根据具体情况选择适合的实现方式,并考虑线程安全性、效率等问题。

在单线程中,这两种方式都可以使用。然而,在多线程环境下,懒汉式单例模式可能会引发线程安全问题,因为多个线程可能同时访问并创建该单例对象。为了解决这个问题,可以使用以下两种方式:

  1. 加锁:在getInstance函数中使用互斥锁来保证线程安全。

  2. 双检查锁:在getInstance函数中使用双检查锁机制,先检查instance是否为空,如果为空,则获取互斥锁,再次检查instance是否为空,如果为空,则创建对象。

综上所述,饥汉式单例模式更适合在线程中使用,因为它没有线程安全问题,并且更高效,因为它在程序启动时就已经创建了实例。但是需要注意的是,饥汉式单例模式可能会增加程序启动时间和内存使用。如果单例对象非常庞大,或者不一定在每次都被使用时都需要创建,那么懒汉式单例模式可能更适合使用。

#include <iostream>
#include <thread>
#include <mutex>class Singleton {
private:static Singleton* instance;static std::mutex mutex;Singleton() {}
public:static Singleton* getInstance() {std::lock_guard<std::mutex> lock(mutex);if (instance == nullptr) {instance = new Singleton();}return instance;}void showMessage() {std::cout << "Hello from Singleton!" << std::endl;}
};Singleton* Singleton::instance = nullptr;
std::mutex Singleton::mutex;void threadFunction() {Singleton* singleton = Singleton::getInstance();singleton->showMessage();
}int main() {const int numThreads = 5;std::thread threads[numThreads];for (int i = 0; i < numThreads; ++i) {threads[i] = std::thread(threadFunction);}for (int i = 0; i < numThreads; ++i) {threads[i].join();}return 0;
}

当在多线程环境中使用单例模式时,需要考虑线程安全性以避免多个线程同时创建实例或访问共享资源的问题。以下是一个示例,在多线程环境下使用懒汉式单例模式,并通过加锁确保线程安全:。。。。

在上面的示例中,我们使用了 std::mutex 来确保在 getInstance() 函数中对单例对象进行加锁操作,从而保证多线程环境下的线程安全性。每个线程调用 getInstance() 函数来获取单例实例,并调用 showMessage() 方法输出信息。

这样,在多线程环境下,每个线程都会获得同一个实例,并且不会出现多个实例被创建的情况。通过加锁操作,确保了线程安全性。

#include <iostream>class Singleton {
public:static Singleton& getInstance() {static Singleton instance; // 在首次调用时初始化静态局部变量return instance;}void showMessage() {std::cout << "Hello, I am a singleton instance!" << std::endl;}private:Singleton() {} // 将构造函数私有化,防止外部实例化对象Singleton(const Singleton&) = delete; // 禁用拷贝构造函数Singleton& operator=(const Singleton&) = delete; // 禁用赋值运算符
};int main() {Singleton& singleton = Singleton::getInstance();singleton.showMessage();// 以下代码会报错,因为构造函数是私有的// Singleton newSingleton;// Singleton anotherSingleton = singleton;return 0;
}
class Singleton {
private:static Singleton* instance;Singleton(){}
public:static Singleton* getInstance(){if(instance == nullptr){instance = new Singleton();}return instance;}
};Singleton* Singleton::instance = nullptr;

http://www.lryc.cn/news/306056.html

相关文章:

  • JavaSec 基础之 XXE
  • AI:137-基于深度学习的实时交通违法行为检测与记录
  • 【深度学习笔记】3_14 正向传播、反向传播和计算图
  • Jenkins详解
  • Java8 Stream API 详解:流式编程进行数据处理
  • 【机器学习】特征选择之过滤式特征选择法
  • C#_扩展方法
  • LeetCode 热题 100 | 二叉树(一)
  • k8s之nodelocaldns与CoreDNS组件
  • Java中的访问修饰符
  • 【论文解读】transformer小目标检测综述
  • springboot215基于springboot技术的美食烹饪互动平台的设计与实现
  • Rust核心:【所有权】相关知识点
  • 单片机05__串口USART通信__按键控制向上位机传输字符串
  • 实习日志30
  • 【MySQL】探索表结构、数据类型和基本操作
  • 解决采集时使用selenium被屏蔽的办法
  • stream流-> 判定 + 过滤 + 收集
  • 人工智能在测绘行业的应用与挑战
  • 四、分类算法 - 随机森林
  • pytorch -- DataLoader
  • 【MySQL面试复习】索引创建的原则有哪些?
  • 四种主流的prompt框架
  • Educational Codeforces Round 160 (Rated for Div. 2) E. Matrix Problem(费用流)
  • 基于SpringBoot的气象数据监测分析大屏
  • 关于硅的制造芯片的过程
  • 【深度学习笔记】3_10 多层感知机的PyTorch实现
  • 输入法在 Android13上候选词 候选区域 不显示的问题
  • Java 面向对象进阶 18 JDK8、9开始新增的方法;接口的应用;适配器设计模式;内部类(黑马)
  • 数据结构-二分搜索树(Binary Search Tree)