设计模式——单例模式

一、单例模式

单例模式是什么?

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

应用场景

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

实现单例模式的关键

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

单例模式类型

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

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

实现一个单例类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变量是否已经初始化,如果没有初始化就会进行初始化,这个判断逻辑会消耗一点性能)

相关推荐
林开落L4 小时前
线程进阶:线程池、单例模式与线程安全深度解析
linux·安全·单例模式·线程池
岁岁岁平安5 小时前
Java的双重检查锁机制(DCL)与懒加载的单例模式
java·单例模式·synchronized·
Deschen5 小时前
设计模式-工厂模式
设计模式·简单工厂模式
阿无,5 小时前
Java设计模式之工厂模式
java·开发语言·设计模式
Camel卡蒙7 小时前
DDD架构——充血模型、领域模型
java·设计模式·架构
rongqing20197 小时前
Google 智能体设计模式:目标设定与监控
设计模式
weixin_445476688 小时前
一天一个设计模式——开闭原则
服务器·设计模式·开闭原则
李广坤8 小时前
模板方法模式(Template Method Pattern)
设计模式
Asort15 小时前
JavaScript设计模式(十二)——代理模式 (Proxy)
前端·javascript·设计模式
简小瑞15 小时前
VSCode源码解密:Event<T> - 类型安全的事件系统
前端·设计模式·visual studio code