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

【单例模式】饿汉式与懒汉式以及线程安全

1. 单例模式介绍

饿汉式单例模式:还没有获取实例对象,实例对象就已经产生了。一定是线程安全的。

懒汉式单例模式:需要用的时候再构造实例。

应用场景:比如日志模块,数据库模块,开发的解析器模块。

2.饿汉式单例实现

2.1构造步骤

1. 构造函数私有化

2. 定义一个唯一的类的静态实例对象(类内声明类外初始化)

3. 获取类的唯一实例对象的接口方法

4. 删除拷贝构造和赋值函数

// 饿汉式单例模式
class Singleton
{
public:static Singleton* getInstance() // 3. 获取类的唯一实例对象的接口方法{return &instance;}private:static Singleton instance;  // 2. 定义一个唯一的类的实例对象Singleton() // 1. 构造函数私有化{}// 4. 删除拷贝构造 和 赋值函数Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
};
Singleton Singleton::instance;  // 类内声明类外初始化

2.2饿汉式单例特点

饿汉式单例模式一定是线程安全的。

缺点:对象的实例化构造中可能需要做很多东西,程序中用不到也会执行,程序运行前耗费太多时间。希望在需要用的时候再构造实例。

3.懒汉式单例实现

3.1懒汉式单例实现步骤

就是将饿汉式单例的实例对象定义为指针。

// 懒汉式单例模式 ==> 是不是线程安全呢?
class Singleton
{
public:// 是不是可重入函数呢?static Singleton* getInstance() // 3. 获取类的唯一实例对象的接口方法{if(instance == nullptr){/*1 开辟内存2 给instance赋值3 构造对象  */instance = new Singleton();}return instance;}private:static Singleton *instance;  // 2. 定义一个唯一的类的实例对象Singleton() // 1. 构造函数私有化{}// 4. 删除拷贝构造 和 赋值函数Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
};
Singleton *Singleton::instance = nullptr;  // 类内声明类外初始化

3.2懒汉式单例特点

不是一个线程安全的:编译器也许会指令优化如2 3顺序可能不同 ==> 多个线程可能会出问题 ==> 不是可重入函数,使用锁实现互斥。

4.实现线程安全的单例模式

4.1实现思路

  1. lock_gard放在if上面会导致锁的粒度太大
  2. 放在if里面要注意使用“锁+双重判断”(防止两个线程都进入if构造两个实例)。
  3. instance在数据段,由统一进程内多个线程共享,cpu为了加快指令执行,会让线程共享的内存值都拷贝一份放自己缓存里,需要给指针加关键字volatile。好处是:当一个线程给instance赋值的时候,其他线程马上就能看到instance改变,因为线程已经不对这个共享变量进行缓存了。

线程安全的懒汉式单例模式:

// 懒汉式单例模式 ==> 是不是线程安全呢?  ==> 线程安全的懒汉式单例模式
std::mutex mtx;
class Singleton
{
public:// 是不是可重入函数呢?     锁+双重判断static Singleton* getInstance() // 3. 获取类的唯一实例对象的接口方法{// std::lock_guard<std::mutex> guard(mtx); // 锁的粒度太大了if(instance == nullptr){std::lock_guard<std::mutex> guard(mtx);if(instance == nullptr){/*1 开辟内存2 给instance赋值3 构造对象  编译器也许会指令优化如2 3顺序可能不同 ==> 多个线程可能会出问题 ==> 不是可重入函数,使用锁实现互斥。*/instance = new Singleton();}}return instance;}private:static Singleton *volatile instance;  // 2. 定义一个唯一的类的实例对象Singleton() // 1. 构造函数私有化{}// 4. 删除拷贝构造 和 赋值函数Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
};
Singleton *volatile Singleton::instance = nullptr;  // 类内声明类外初始化

4.2更精简的线程安全的懒汉式单例模式

参考文献列在最后了!

// 线程安全的懒汉式单例模式(更精简)
class Singleton
{
public:static Singleton* getInstance() // 3. 获取类的唯一实例对象的接口方法{// 静态的局部变量,程序开始内存就有了,在数据段上。// 静态对象在第一次运行到的时候才进行初始化。没运行getInstance就不会构造实例。// g++ -g -o run singleton.cpp  gdb run// 函数静态局部变量的初始化,在汇编指令上已经自动添加线程互斥指令了。static Singleton instance;  // 2. 定义一个唯一的类的实例对象。return &instance;}private:Singleton() // 1. 构造函数私有化{}// 4. 删除拷贝构造 和 赋值函数Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
};

这种构造使用静态局部变量的特点:

        // 静态的局部变量,程序开始内存就有了,在数据段上。
        // 静态对象在第一次运行到的时候才进行初始化。没运行getInstance就不会构造实例。

通过

         // g++ -g -o run singleton.cpp  gdb run
        // 函数静态局部变量的初始化,在汇编指令上已经自动添加线程互斥指令了。

在Linux环境中,通过g++编译上面的代码,命令如下:
g++ -o main main.cpp -g
生成可执行文件main,用gdb进行调试,到getInstance函数,并打印该函数的汇编指令,如下:

可以看到:对于static静态局部变量的初始化,编译器会自动对它的初始化进行加锁和解锁控制,使静态局部变量的初始化成为线程安全的操作,不用担心多个线程都会初始化静态局部变量,因此上面的懒汉单例模式是线程安全的单例模式!

参考博客:C++设计模式 - 单例模式_大秦坑王 设计模式-CSDN博客


http://www.mrgr.cn/news/67524.html

相关文章:

  • A013-基于SpringBoot的宽带业务管理系统的设计与实现
  • Apache-Hive数据库使用学习
  • 算法工程师重生之第四十五天(孤岛的总面积 沉没孤岛 水流问题 建造最大岛屿)
  • 利用全排列解决LeetCode第3343题“统计平衡排列的数目”问题
  • 【网络安全】|nessus使用
  • 免费,基于React + ECharts 国产开源 IoT 物联网 Web 可视化数据大屏
  • 嵌入向量模型与BM25算法结合:并行检索获取多种结果
  • 常见几种GB 9706.1-2020医疗器械试验工装,您有所了解吗?
  • 使用stream遍历对象集合,取出所有对象的某字段,并以逗号拼接起来
  • 【TabBar嵌套Navigation案例-常见问题按钮-WebView-加载JavaScript Objective-C语言】
  • 杭州电商运营公司排名:怎么找到适合自己的电商代运营公司
  • Java基础知识
  • 模拟计算机如何识别和执行机器语言指令:从虚拟CPU的角度解析
  • C#-哈希表
  • 七载同行,共襄盛会!苏州金龙高标准服务进博会彰显中国智造风采
  • Java手写二分查找
  • 部署一个属于自己的文件服务器(File Browser )
  • 使用亚马逊 S3 连接器为 PyTorch 和 MinIO 创建地图式数据集
  • Halcon 算法加速的基础知识(多核并行/GPU)
  • 多任务学习在转转主搜精排的应用
  • 深圳新世联:氢能中的气体传感器应用
  • 可视化建模与UML《顺序图实验报告》
  • 【青牛科技】D1084 5A低压差电压调整器应用方案
  • 浅谈Spring MVC
  • 6.qsqlquerymodel源码分析
  • Python实现SSA智能麻雀搜索算法优化BP神经网络分类模型(优化权重和阈值)项目实战