设计模式——单例模式

一、单例模式

单例模式是什么?

单例模式是一种创建型的软件设计模式,在工程项目中十分常见。通过单例模式的设计,使得创建的类在当前进程中只有一个实例,并提供一个全局性的访问点,这样可以规避因频繁创建对象而导致的内存飙升情况。

应用场景

有些对象控制大量的共享资源(如数据库或者文件)的访问权限,如果同时存在大量对象进行访问,可能会造成混乱引起安全问题,有些对象本身占用大量的资源,存在过多实例会造成资源的浪费,如果应用中某个类对于所有使用者而言只有一个可用的实例则可使用单例模式。

实现单例模式的关键

  • 私有化构造函数:由于外部无法调用类的私有成员,因此可以阻止用户创建多个实例
  • 移除拷贝构造函数和赋值构造函数
  • 静态私有成员变量:该变量指向生成的唯一实例
  • 公有的静态成员函数:提供外部访问唯一实例的接口

单例模式类型

按唯一实例创建的时机可分为懒汉式和饿汉式两种单例模式

  • 懒汉式: 在需要使用该实例时才去真正的创建该实例,由于是非必要不创建,因此被称为懒汉式。
  • 饿汉式: 在类装载时就进行了实例的创建,就算后面全程没用使用也会创建该实例,有种不管三七二十一先创建再说的意味,因此称之为饿汉式。

实现一个单例类Singleton

基础版本:

私有构造函数:

cpp 复制代码
Private:
Single(){
	std::cout << "调用构造函数" << std::endl
}
~Single(){
	std::cout << "调用析构函数" << std::endl
}

移除拷贝构造函数和赋值构造函数

cpp 复制代码
Singleton(const Singleton&) = delete;
Singleton& operator=(const Singleton&) = delete;

静态私有成员变量指向唯一实例

cpp 复制代码
Static Singleton* instance_;

提供外部访问唯一实例的接口

懒汉式实现方式

基础版本:

cpp 复制代码
static Singleton* GetInstance(){
	if ( instance_ == nullptr){
		instance_ = new Singleton();
	}
	return instance_;
}

当在多线程模式下多个线程同时调用Singleton::GetInstance()时可能会创建多个实例而导致内存泄漏,简单的处理方式是通过互斥锁实现线程安全

cpp 复制代码
static Singleton* GetInstance(){
	if ( instance_ == nullptr){
		m_mutex_.lock();
		if ( instance_ == nullptr){
			instance_ = new Singleton();
		}
		m_mutex_.unlock();
	}
	return instance_;
}
Static std::mutex m_mutex_;

饿汉式实现方式

cpp 复制代码
static Singleton* GetInstance(){
	return instance_;
}
全局变量区
Singleton* Singleton::instance_ = new Singleton();

Meyers' Singleton

cpp 复制代码
static Singleton* GetInstance(){
	static Singleton instance;
	return instance;
}

优点:

● 解决了普通单例模式全局变量初始化依赖(C++只能保证在同一个文件中声明的static遍历初始化顺序和其遍历声明的顺序一致,但是不能保证不同文件中static遍历的初始化顺序)

缺点:

● 需要C11支持(C11保证static成员初始化的线程安全)

● 性能问题(同懒汉模式一样,每次调用GetInstance()方法时需要判断局部static变量是否已经初始化,如果没有初始化就会进行初始化,这个判断逻辑会消耗一点性能)

相关推荐
咖啡八杯15 小时前
GoF设计模式——备忘录模式
java·后端·spring·设计模式
槑有老呆19 小时前
从 Prompt Engineering 到 Harness Engineering:AI 编程的下一次跃迁
设计模式
HjhIron1 天前
从Prompt到Context:大模型应用开发的范式转移
设计模式·aigc·ai编程
咖啡八杯3 天前
GoF设计模式——中介者模式
java·后端·spring·设计模式
胡萝卜术3 天前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
亦暖筑序4 天前
Java 8老系统Browser Agent实战:三层拦截把AI操作后台变成可审计流程
java·后端·设计模式
青禾网络6 天前
Web 前端如何接入 AI 音效生成:从零到可用的完整方案
人工智能·设计模式
ZJPRENO7 天前
吃透软件开发六大设计原则,告别烂代码
设计模式
咖啡八杯7 天前
GoF设计模式——命令模式
java·设计模式·架构
花椒技术8 天前
HJPusher / HJPlayer SDK 实践:我们为什么把直播推播链路拆成一套可复用能力
设计模式·harmonyos·直播