首页 > 编程语言 >【C++设计模式】之单例模式,一文看懂

【C++设计模式】之单例模式,一文看懂

时间:2024-10-27 19:19:53浏览次数:3  
标签:Singleton C++ 实例 线程 模式 单例 设计模式 懒汉

【C++设计模式】之单例模式

一、什么是单例模式

单例模式是一种创建型设计模式,它的作用是确保一个类只有一个实例。这是设计模式中最简单的一种,也是面试时最容易被考到手写代码的一种设计模式。

二、单例模式的实现

一般情况下单例模式有三种实现方式,分别为懒汉版饿汉版线程安全下的懒汉版。我们分别来进行介绍和实现。
单例模式的实现可以总结成一句话:私有的构造函数+公有的静态成员

1. 懒汉版单例模式

何为懒汉版呢?
只有在需要使用对象的时候才去实例化单例对象。这种方式要考虑线程安全,在这里,这个简单的懒汉版先不考虑线程安全问题。
我把一些需要注意的点直接以注释的形式写在代码段中,方便观看。

懒汉单例模式代码实现

# include<iostream>
using namespace std;
//1. 懒汉版(线程不安全)
class Singleton
{
private:
	//私有构造函数防止外部实例化
	Singleton()
	{
		cout << "Singleton Constructor" << endl;
	}
	//禁止拷贝构造和重载赋值,以免拷贝构造生成多个实例
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
	//私有的静态指针 --- 持有唯一实例,因为是静态,所以不属于任何一个类,所以确保了在所有对象间共享并保证只有一个
	static Singleton* instance;
public:
	//公有的静态成员函数 --- 获取单例模式实例的方法
	static Singleton* Get_instance()
	{
		if (instance == nullptr)
		{
			instance = new Singleton();
		}
		return instance;
	}
	void Show()
	{
		cout << "hello" << endl;
	}
};
//静态成员变量只能在类外初始化
Singleton* Singleton::instance = nullptr;
int main()
{
	Singleton* p = nullptr;
	Singleton* p1 = nullptr;
	//公有的静态成员函数是获取单例模式的方法,只能用这种方式来实例化对象
	p = Singleton::Get_instance();
	p1 = Singleton::Get_instance();
	p->Show();
	p1->Show();
	if (p1 == p)
	{
		cout << "相同实例" << endl;
	}
	else
	{
		cout << "不同实例" << endl;
	}
}

在这里插入图片描述
我们可以从编译结果看出来,两次Get但是只构造了一次,说明只有一个实例。

2. 饿汉单例模式

何为饿汉模式?
它在程序运行的那一刻就创建了实例,这本身就是一种线程安全的版本。
但是只要程序运行它就实例化,意味着我们如果用不到这个实例,我们还是创建了实例,这样就造成了资源的浪费。

饿汉单例模式代码实现

# include<iostream>
using namespace std;
//饿汉版 -- 线程安全版本
class Singleton
{
private:
	Singleton()
	{
		cout << "饿汉单例构造函数" << endl;
	}
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
	static Singleton* instance;
public:
	static Singleton* Get_instance()
		//instance不可能为空了,因为饿汉版本直接就在开始就已经实例化了
	{
		return instance;
	}
	void Show()
	{
		cout << "Hello World" << endl;
	}
};
Singleton* Singleton::instance = new Singleton();
int main()
{
	Singleton* p1 = Singleton::Get_instance();
	Singleton* p2 = Singleton::Get_instance();
	p1->Show();
	p2->Show();
	if (p1 == p2)
	{
		cout << "相同实例" << endl;
	}
	else 
	{
		cout << "不同实例" << endl;
	}
}

在这里插入图片描述
从编译结果我们能得出两次想要实例化,但是只有一个实例。

饿汉版和懒汉版代码看上去极为相似,它们的区别在于饿汉在类外初始化静态成员变量的时候直接创建实例,并且在公有的静态成员函数(Get_instance)的时候只返回已经实例了的instance

3. 线程安全的懒汉单例模式

何为线程安全的懒汉模式?
在创建实例的函数中加锁,加锁再做一次判定,防止在多线程创建的时候多个线程都判定没有创建好实例,然后同时创建了实例。

线程安全的懒汉模式代码实现

# include<iostream>
# include<thread>
# include<mutex>

using namespace std;
class Singleton
{
private:
	Singleton()
	{
		cout << "懒汉线程安全 构造函数" << endl;
	}
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
	static Singleton* instance;
	//静态互斥锁,用于线程安全
	static mutex mtx;
public:
	static Singleton* Get_instance()
	{
		if (instance == nullptr)
		{
			//加锁再做一次判定,防止在多线程创建的时候多个线程都判定没有创建好实例,然后同时创建了实例
			lock_guard<mutex> lock(mtx);//加锁
			if (instance == nullptr)
			{
				instance = new Singleton();
			}
		}
		return instance;
	}
};
Singleton* Singleton::instance = nullptr;
mutex Singleton::mtx;
void CreateInstance()
{
	Singleton* p = nullptr;
	p = Singleton::Get_instance();
	cout << p << endl;
}
int main()
{
	thread t1(CreateInstance);
	thread t2(CreateInstance);
	t1.join();
	t2.join();
}

在这里插入图片描述
我们能从结果看出来,两个线程实例的地址相同,意味着还是单例模式。
注意:
线程安全的懒汉模式和线程不安全的懒汉模式的区别在于,线程安全的懒汉模式用到了互斥锁,并且这个锁是私有静态,而且在进行实例的时候进行了两次判断,加锁后进行判断确保了线程安全。

三、总结

  • 单例模式确保一个类只有一个实例,并且有一个全局访问点。
  • 它有三种实现模式,分别为懒汉模式,饿汉模式和线程安全的懒汉模式。
  • 实现单例模式可以总结成一句话:私有的构造函数和公有的静态全局函数。
  • 在声明单例类的时候,定义为私有的有:构造函数,禁止拷贝构造和赋值重载,静态指针(保存唯一实例)。声明为公有的有:静态的全局函数(获取唯一的实例的方法)。

标签:Singleton,C++,实例,线程,模式,单例,设计模式,懒汉
From: https://blog.csdn.net/weixin_61276540/article/details/143271181

相关文章

  • 探索C++的奥秘之C/C++内存管理
    一个程序当中的数据主要有以下几类:局部数据、静态数据和全局数据、常量数据、动态申请数据。1.C/C++内存分布1.栈又叫堆栈--非静态局部变量/函数参数/返回值等等,栈是向下增长的。2.内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系......
  • 编译器的实现是用C好还是C++
    标题:编译器的实现是用C好还是C++在探讨编译器的实现时,是否选择C语言或C++取决于多个因素,包括性能需求、团队熟悉度、以及项目的特定需求。C语言以其运行时性能和底层操作的直接性著称,是许多传统编译器如GCC的选择。C++提供了更高级的抽象和面向对象的特性,使得大型项目的组织和维......
  • c++第6课
    1.c++第6课今天我们来学逻辑运算符,即||&&!!||的意思是或&&的意思是与!!的意思是非因此我们可以和第5课ifelse一起来写#include<bits/stdc++.h>usingnamespacestd;intmain(){inta; cin>>a; if(a%5==0||a%7==0)cout<<"yes";//这里用了或|| elsecout<......
  • 关于C++游戏开发
            C++是一种广泛用于游戏开发的编程语言,尤其是在性能要求较高的游戏中。它提供了对硬件级别的控制以及高效的执行速度,这使得C++成为了许多专业游戏引擎和AAA级游戏的选择。以下是一些关于C++游戏开发的关键点:1.游戏引擎UnrealEngine:使用C++作为主要编......
  • C++——多态
    1.概念多态:通俗来说,就是多种形态。具体说就是去完成某个行为时,不同的对象去完成会产生出不同的状态,这就是多态。比如,同样是买票,当普通人买票时,是全价买票;而学生买票时,是半价买票;军人买票时,是优先买票。2.构成条件C++里,在继......
  • Chromium base库 环境变量类使用说明c++
    1、环境变量获取和设置定义参考:base\environment.hbase\environment.cc//Copyright2011TheChromiumAuthors//UseofthissourcecodeisgovernedbyaBSD-stylelicensethatcanbe//foundintheLICENSEfile.#ifndefBASE_ENVIRONMENT_H_#defineBASE_EN......
  • VC++ __declspec(dllexport) 和 __declspec(dllimport)
    头文件中声明了方法,在提供者那里方法应该被声明为__declspec(dllexport),在使用者那里,方法应该被声明为__declspec(dllimport)。Class中含有一个静态变量,生成dll的时候只采用了__declspec(dllexport),使用的时候__declspec(dllimport)就派上用场了,他会告诉使用dll的工程去lib中找......
  • C++/C电子宠物1.0
    使用的是一个在线编程的网站,是linux环境https://www.onlinegdb.com/#include<iostream>#include<stdlib.h>#include<time.h>#include<unistd.h>#include<pthread.h>#include<string>usingnamespacestd;intwindow();intea......
  • 【C++ 真题】B2099 矩阵交换行
    矩阵交换行题目描述给定一个5×55\times55×5的矩阵(数学上,一个......
  • C++ 静态变量什么时候完成初始化
    C语言在编译器就完成静态变量的内存分配和初始化;始化发生在任何代码执行之前,属于编译期初始化。C++全局或静态对象当且仅当对象首次用到时才进行构造,并通过atexit()来管理对象的生命期;静态变量初始化是线程安全的。全局变量、文件域的静态变量和类的静态成员变量在m......