C++ 单例模式

单例模式

  • 一个类仅有一个实例
  • 提供该实例的全局访问点

单线程下的单例模式:

cpp 复制代码
class Singleton{
public:
	static Singleton * GetInstance(){
		if(_instance == nullptr)
		{
			_instance = new Singleton();
			std::atexit(Destructor);
		}
		return _instance;
	}
private:
	static void Destructor(){
		if(_instance != nullptr){
			delete _instance;
			_instance = nullptr;
		}
	}
	Singleton() {}
	~Singleton() {}
	Singleton(const Singleton &clone) {}
	Singleton & operator=(const Singleton &) {};
	static Singleton *_instance;
};
Singleton *Singleton::_instance = nullptr;

1.静态成员函数和静成员变量

2.构造、析构、赋值操作符、拷贝构造私有化

3.new是堆上创建std::atexit

多线程下的单例模式:

cpp 复制代码
#include <mutex>
#include <atomic>
class Singleton{
public:
	static Singleton * GetInstance(){
		Singleton *temp = _instance.load(std::memory_order_relaxed);
		std::atomic_thread_fence(std::memory_order_acquire);//内存栅栏,确保在此点之前的所有读操作都完成,并且在此点之后的所有读操作都不会被重排到这个点之前
		//std::lock_guard<std::mutex> lock(_mutex);
		if(_instance == nullptr)
		{
			std::lock_guard<std::mutex> lock(_mutex);
			temp = _instance.load(std::memory_order_relaxed);
			if(_instance == nullptr)
			{
				temp = new Singleton();//多线程环境下编译器和cpu对程序进行优化
				std::atomic_thread_fence(std::memory_order_release);
				_instance.store(temp,std::memory_order_relaxed);
				//1.分配内存 2.调用构造函数 3.返回指针、赋值运算
				//优化为1、3、2
				std::atexit(Destructor);
			}		
		}
		return _instance;
	}
private:
	static void Destructor(){
		Singleton *temp = _instance.load(std::memory_order_relaxed);
		if(temp != nullptr){
			delete temp;
			_instance.store(temp,std::memory_order_relaxed);
			temp = nullptr;
		}
	}
	Singleton() {}
	~Singleton() {}
	Singleton(const Singleton &clone) {}
	Singleton & operator=(const Singleton &) {};
	static std::atomic<Singleton *> _instance;
	static std::mutex _mutex;
};
std::atomic<Singleton *> Singleton::_instance = nullptr;
std::mutex Singleton::_mutex;

1.两次判断,希望在第二次判断的时候加互斥锁

2.多线程环境下,编译器、cpu的优化(指令重排)------原子操作+内存屏障


获取更多Linux C/C++资料

相关推荐
用户805533698031 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
BadBadBad__AK2 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境2 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境2 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴3 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
卷无止境5 天前
C++ 的Eigen 库全解析
c++
卷无止境5 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴5 天前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake
博客18007 天前
酷宝的使用方法,超好用的免费界面库,C++、MFC可用
c++·mfc·界面库·库来帮·酷宝
郝学胜_神的一滴7 天前
CMake 026:属性体系精讲、四大作用域全解 & 实战代码落地
c++·cmake