C++实现单例模式

单例模式是一种创建型模式,核心思想是保证一个类只有一个实例,并对外提供一个全局访问点来访问这个实例

单例模式分为两种:

  • 饿汉式:指的是在程序一加载就创建实例,因此称为饿汉式。这样的好处是没有多线程竞争,缺点是如果后面没有使用到这个单例,就浪费了。
  • 懒汉式:指的是到要用的时候才创建,因此称为懒汉式。这种方式可能会有多线程冲突,因此程序运行中,可能有多个线程都想创建单例,因此需要使用相应的机制避免冲突。程序需要的时候才进行实例化,提高了性能。

单例模式的优点:

  1. 全局控制:保证只有一个实例,就可以严格控制什么时候如何访问
  2. 节省资源:因为只有一个实例,不会创建多个实例,自然就更节省资源了

单例模式实现的几个要点:

  • 私有的构造函数:防止外部直接创建
  • 私有的静态实例变量:保存唯一实例
  • 公有的访问方法:对外的唯一方法来获取实例

下面是一个常见的懒汉式实现方式:

cpp 复制代码
// singleton.h

#pragma once
#include <mutex>
using namespace std;

class Singleton {
private:
	Singleton() = default;  // 私有构造,禁止外部构造
	~Singleton() = default; // 私有析构,禁止外部析构
	Singleton(const Singleton&) = delete; // 删除拷贝构造函数
	Singleton& operator=(const Singleton&) = delete; // 删除赋值函数
public:
	// 外部接口,获取单例实例对象
	static Singleton* getInstance(); 

	// 释放单例
	static void deleteInstance();

	void print();
private:
	static Singleton* instance; // 唯一的单例对象指针
	static mutex mtx;
};
cpp 复制代码
// singleton.cpp
#include <iostream>
#include "singleton.h"

Singleton* Singleton::instance = nullptr;
mutex Singleton::mtx;
Singleton* Singleton:: getInstance() {
	// 这两个if称为双重检测
	if (instance == nullptr) {
		unique_lock<mutex> lck(mtx);
		if (instance == nullptr) {
			instance = new Singleton();
			return instance;
		}
	}
	return instance;
}

void Singleton::deleteInstance() {
	unique_lock<mutex> lck(mtx);
	if (instance != nullptr) {
		delete instance;
		instance = nullptr;
	}
}

void Singleton::print() { std::cout << "实例地址:" << this << std::endl; }
cpp 复制代码
int main() {
	Singleton::getInstance(); // 获取单例
	Singleton::getInstance()->print();
	return 0;
}

为什么要使用双重检测(上面的两个if)

第一个if很好理解,当实例为空的时候才去创建。第二个if是为了保证只有一个线程拿到锁去创建实例。通俗的说,当进入第一个if后,这时候可能有多个线程来获取锁,如果不第二次判空,就可能创建出多个实例。

饿汉式实现:

cpp 复制代码
class Singleton {
private:
	Singleton() = default;  // 私有构造,禁止外部构造
	~Singleton() = default; // 私有析构,禁止外部析构
	Singleton(const Singleton&) = delete; // 删除拷贝构造函数
	Singleton& operator=(const Singleton&) = delete; // 删除赋值函数
public:
	// 外部接口,获取单例实例对象
	static Singleton* getInstance() {
		return instance;
	}
	void deleteInstance() {
		if (instance != nullptr) {
			delete instance;
			instance = nullptr;
		}
	}
	void print() { std::cout << "实例地址:" << this << std::endl; }
private:
	static Singleton* instance;
};
// 程序一启动就创建,没有多线程冲突
Singleton* Singleton::instance = new Singleton();

int main() {
	Singleton::getInstance();
	Singleton::getInstance()->print();
	return 0;
}

更多的单例模式实现方式可参考:
https://blog.csdn.net/unonoi/article/details/121138176

相关推荐
雍凉明月夜10 分钟前
c++ 精学笔记记录Ⅲ
c++·笔记·学习
oioihoii24 分钟前
C++共享内存小白入门指南
java·c++·算法
布茹 ei ai25 分钟前
QtWeatherApp - 简单天气预报软件(C++ Qt6)(附源码)
开发语言·c++·qt·开源·开源项目·天气预报
Bruce_kaizy26 分钟前
c++图论————图的基本与遍历
c++·算法·图论
Zmm147258369_28 分钟前
好用的PC耐力板机构
c++
Code Slacker1 小时前
LeetCode Hot100 —— 普通数组(面试纯背版)(五)
数据结构·c++·算法·leetcode·面试
秦苒&1 小时前
【C语言】详解数据类型和变量(一):数据类型介绍、 signed和unsigned、数据类型的取值范围、变量、强制类型转换
c语言·开发语言·c++·c#
清水白石0081 小时前
《用 Python 单例模式打造稳定高效的数据库连接管理器》
数据库·python·单例模式
智者知已应修善业2 小时前
【删除有序数组中的重复项 II之O(N)算法】2024-1-31
c语言·c++·经验分享·笔记·算法
爱装代码的小瓶子2 小时前
【c++进阶】C++11新特性:一切皆可{}初始化
开发语言·c++·visual studio