一、单例模式
单例模式是什么?
单例模式是一种创建型的软件设计模式,在工程项目中十分常见。通过单例模式的设计,使得创建的类在当前进程中只有一个实例,并提供一个全局性的访问点,这样可以规避因频繁创建对象而导致的内存飙升情况。
应用场景
有些对象控制大量的共享资源(如数据库或者文件)的访问权限,如果同时存在大量对象进行访问,可能会造成混乱引起安全问题,有些对象本身占用大量的资源,存在过多实例会造成资源的浪费,如果应用中某个类对于所有使用者而言只有一个可用的实例则可使用单例模式。
实现单例模式的关键
- 私有化构造函数:由于外部无法调用类的私有成员,因此可以阻止用户创建多个实例
- 移除拷贝构造函数和赋值构造函数
- 静态私有成员变量:该变量指向生成的唯一实例
- 公有的静态成员函数:提供外部访问唯一实例的接口
单例模式类型
按唯一实例创建的时机可分为懒汉式和饿汉式两种单例模式
- 懒汉式: 在需要使用该实例时才去真正的创建该实例,由于是非必要不创建,因此被称为懒汉式。
- 饿汉式: 在类装载时就进行了实例的创建,就算后面全程没用使用也会创建该实例,有种不管三七二十一先创建再说的意味,因此称之为饿汉式。
实现一个单例类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变量是否已经初始化,如果没有初始化就会进行初始化,这个判断逻辑会消耗一点性能)