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

C++多线程编程——call_once和单例模式

目录

1. 前言

2. call_once和once_flag

3. 后记

3.1 单例类的析构问题

3.2 饿汉式单例模式的线程安全问题


1. 前言

之前在讲解单例模式时,有提到懒汉式单例模式使用了双重检测Double-Checked Locking Pattern (DCLP)来解决多线程的安全访问问题。但是该方法也存在安全隐患

双重检查之所以有问题,是因为CPU会进行指令重排序。
instance = new Singleton;这条语句 一般会理解为构造一个对象并初始化后,然后赋值给instance 。
但是实际上CPU有可能先构造一个空对象 ,把这个空对象的地址赋值给instance , 最后才调用构造函数进行初始化。如果在调用构造函数对这片内存进行初始化之前发生了线程切换,另一个线程检查instance发现不为nullptr,进而使用instance,就会导致程序崩溃。

2. call_once和once_flag

在C++11中提供了call_once和once_flag,通过它们的配合使用,可以保证在多线程环境下某个可调用对象只执行一次。

这样,我们就可以把instance的初始化单独放到一个静态函数中,并通过call_once来执行。

具体请看以下代码:

#include <iostream>
#include <thread>


using namespace std;

class Singleton
{
private:
	static Singleton* instance;
	static once_flag init_flag;
	Singleton() = default;
	static void init_instance()
	{
		instance = new Singleton();
	}
public:
	~Singleton() = default;
	static Singleton* getInstance()
	{
		call_once(init_flag, &Singleton::init_instance);
		return instance;
	}
};

Singleton* Singleton::instance = nullptr;
once_flag Singleton::init_flag;

int main()
{
	Singleton* s1 = Singleton::getInstance();
}

3. 后记

3.1 单例类的析构问题

关于单例类的析构问题,可以采用之前介绍过的嵌套类的方式实现。也可以采用智能指针的方式实现,把instance类型改成shared_ptr<Singleton>,在静态对象消亡时,引用计数归零,会自动调用析构函数。

3.2 饿汉式单例模式的线程安全问题

在C++11之后,静态对象和全局对象的初始化一定是线程安全的,所以可以放心地使用。


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

相关文章:

  • 拍照对比,X70 PRO与X90 PRO+的细节差异
  • 【自然语言处理(NLP)】NLP实战:IMDB影评情感分析项目
  • 全面解析机器学习优化算法中的进化策略
  • 使用 DeepSeek-R1 与 AnythingLLM 搭建本地知识库
  • 排序算法--插入排序
  • 基序和纯度分数的计算
  • 【AI日记】25.02.05 自由不是一种工具
  • 2025年2月4日--2月9日(ue4.0shader抄写+ue5肉鸽独立游戏视频)
  • DeepSeek大模型介绍、本地化部署与使用!【AI大模型】
  • 数据库系统概念第六版记录 一
  • 【prompt实战】AI +OCR技术结合ChatGPT能力项目实践(BOL提单识别提取专家)
  • 总结11..
  • Vue - customRef 自定义ref
  • shiro面试题
  • 【含文档+PPT+源码】基于Python爬虫二手房价格预测与可视化系统的设计与实现
  • Vue的状态管理:用响应式 API 做简单状态管理、状态管理库(Pinia )
  • 【论文精读】Taming Transformers for High-Resolution Image Synthesis
  • 【入门】如何使用DeepSeek批量创作短视频
  • git 指定ssh key
  • 【漫话机器学习系列】081.测量理论(Almost Everywhere)
  • 3D图形学与可视化大屏:如何让材质与光照进行交互。
  • C# 字符串与正则表达式介绍
  • 2.5学习
  • 阿里 Java 岗个人面经分享(技术三面 + 技术 HR 面):Java 基础 +Spring+JVM+ 并发编程 + 算法 + 缓存
  • 【学Rust写CAD】3 绝对坐标系详解
  • mini-lsm通关笔记Week2Day6