探索设计模式——单例模式详解

news2024/12/28 3:52:15

        前言:设计模式的作用主要是为了——利用设计方式的重用来自动地提高代码的重新利用、提高代码的灵活性、节省时间, 提高开发效率、低耦合,封装特性显著, 接口预留有利于扩展。

        设计模式的种类有很多种,本篇内容主要讲解其中的单例模式。

什么是单例模式

        单例模式是创建型模式的一种。顾名思义, 单例模式就是全局只有一个实例化对象。该种模式可以保证一个类不能创建新的对象。 也就是不能直接定义和new。      单例模式保证了全局只有唯一一份实例化对象, 而且它能够自行实例化并且向整个系统提供这个实例。这种类,提供了全局的方法, 叫做单例类, 而这种设计方法, 叫做单例模式。

        同时, 单例模式又分为饿汉模式和懒汉模式两种。下面为他们两个的特性:

  • 饿汉模式:程序开始前先将唯一实例创建出来。
  • 懒汉模式:第一次使用时再将唯一实例创建出来。

        两者的优缺点:

  • 饿汉模式:优点是设计简单。但缺点是可能导致程序的启动较慢,不能延迟加载。并且当有多个单例对象的时候, 哪个对象先生成不确定。
  • 懒汉模式:优点是解决了程序启动慢的问题, 延迟了加载。 并且当有多个单例对象的时候, 可以创建的顺序。 但缺点是设计比较复杂, 同时存在线程安全问题, 解决线程安全问题会导致程序的性能降低。

        综上, 我们要知道单例模式的三个主要特性: 

  • 一个类里面只能有一份实例。
  • 这个类能够自行创建这个实例。
  • 必须向整个系统提供这个实例。

为什么要有单例模式:

        在我们的电脑中其实有许多程序我们都只能打开一次, 其实这就是应用了单例模式。 比如当我们打开“设置”, 如果你多次点击“设置”。 他也只能打开一个设置窗口(如果你打开了多个,并且不是你本人自己修改了操作系统。那么请给你的电脑杀一下毒)。 我们不妨想一下, 如果我们能够打开多个设置, 那么是不是就显得有点多余, 因为我们只使用一个设置窗口就可以,不需要使用多个。开启多个设置窗口不仅会占用系统上的资源,并且, 如果我们能够打开多个设置窗口。 那么我们在这个窗口设置成这个样子, 在那个窗口设置成那个样子。 最终, 设置的结果是用哪个窗口的呢?所以, 这都是问题。 而单例模式就是为了解决这些的问题。 

代码实现

      饿汉模式

        饿汉模式就是在程序启动之前就自行创建好了这个实例。 这一点我们可以通过静态成员变量来实现, 同时为了保证类的封装性, 我们将这个静态成员变量私有化:

	class Only_Instance 
	{
	private: static Only_Instance _inst;        //私有化唯一实例
	};

    Only_Instance Only_Instance::_inst;         //静态成员变量的定义。

        然后为了保证全局只有唯一一份实例, 我们需要私有化该类的构造函数, 这样就能保证类外不能随意创建对象。 

	class Only_Instance 
	{
	private: static Only_Instance _inst;        //类唯一实例
	//私有化构造函数
	private: Only_Instance() {};                
	private: Only_Instance(const Only_Instance&) = delete;       
	};

	Only_Instance Only_Instance::_inst;         //静态成员变量定义

         如此一来, 我们就能保证:该类只有唯一一份实例化对象。 并且能够在程序启动之前自行创建一份实例。

        那么就要解决访问这个唯一一份实例的问题, 我们如何对它进行访问? 我们可以通过一个共有的方法来访问它, 并且因为我们没有实例化对象调用成员函数以及方法的全局性, 所以这个方法必须是静态的, 全局的:

	class Only_Instance 
	{
	private: static Only_Instance _inst;        //类唯一实例
	//私有化构造函数
	private: Only_Instance() {};                
	private: Only_Instance(const Only_Instance&) = delete;       
	public: 
		static Only_Instance* getInstance()   //公有的全局方法
		{
			return &_inst;
		}
	};

	Only_Instance Only_Instance::_inst;         //静态成员变量定义

这样,我们就设计出了一个简单的饿汉模式。但是饿汉的不能延迟加载, 启动慢是一个问题。 为了解决, 这个问题。 我们这里看一下另一个设计模式——懒汉模式

        懒汉模式

懒汉模式是在第一次使用的时候自行生成实例。 能够延迟加载, 解决了启动慢的问题。

懒汉模式同样需要私有化构造函数, 不能在外部创建实例化对象:

	class Only_Instance 
	{
		//构造函数
	private:Only_Instance() {};
	private:Only_Instance(const Only_Instance&) = delete;

	};

为了能够第一次才实例化对象, 我们需要在类地内部创建对象。 那么就不能定义一个全局的单例对象, 应该定义一个全局的单例对象指针, 并且这个指针也是私有的:

	class Only_Instance 
	{
		//构造函数
	private:Only_Instance() {};
	private:Only_Instance(const Only_Instance&) = delete;
	private:static Only_Instance* _inst;                       //全局的对象指针。

	};

 那么如何才能访问这个指针, 我们用到和饿汉相同的方式。 就是定义一个全局的静态方法:

	class Only_Instance 
	{
		//构造函数
	private:Only_Instance() {};
	private:Only_Instance(const Only_Instance&) = delete;
	private:static Only_Instance* _inst;                       //全局的对象指针。
	public:
		static Only_Instance* getInstance()     //访问inst
		{
			if (_inst == nullptr) 
			{
				_inst = new Only_Instance();
			}
			return _inst;
		}

	};

        这样, 就能创建一个通过受控地_inst, 随时访问我们规定的方法getInstance。

        然后, 为了能够手动的销毁这个实例, 我们还要提供一个自动释放的接口:

		//销毁实例化对象
		public static void DelInstance() 
		{
			delete _inst;
		}

 并且,为了资源能够自动释放。 我们可以定义一个内部类成员。 当这个成员销毁的时候, 那么调用inst的析构, 即:

	class Only_Instance 
	{
		//构造函数
	private:Only_Instance() {};
	private:Only_Instance(const Only_Instance&) = delete;
	private:static Only_Instance* _inst;                       //全局的对象指针。
	public:
		static Only_Instance* getInstance()     //访问inst
		{
				if (_inst == nullptr)
				{
					_inst = new Only_Instance();
				}

			return _inst;
		}
		
		//销毁实例化对象
		static void DelInstance() 
		{
			delete _inst;
		}

		class auto_des
		{
		public:
			~auto_des() 
			{
				delete _inst;
			}
		};
		static auto_des _ad;
	};
	
	Only_Instance::auto_des Only_Instance::_ad;

        但是, 这个懒汉模式此时有另外的问题。 当涉及多线程时,如果我们一个程序跑到了if语段, 另一个程序也跑到了if语段。 那么如果两个if语段同时执行, 那么就会造成创建多个实例对象的情况。 为了避免这种情况,这里就要用到加锁地操作。 这个由于博主知识板块不太完整, 暂时不进行详细讲解。

        想要了解地请看大佬的这篇文章:确保对象的唯一性——单例模式 (三)_青兮科技公司开发人员使用单例模式实现了负载均衡器的设计,但是在实际使用中出现-CSDN博客

其他简单特殊类设计

只能在堆上创建对象

        只能在堆上创建对象, 只能在堆区创建对象。 那么我们就要让这个类无法随便创建对象。 那么使用的方法就是私有化或者封掉构造函数。 即:

class Only_Heap
{
public:
	Only_Heap* CreatObj()
	{
		return new Only_Heap;
	}

	//要注意将拷贝构造禁掉
	Only_Heap(const Only_Heap& pt) = delete;
private:
	//
	Only_Heap()
	{
		cout << "构造对象" << endl;
	}
	int a = 1;
};

只能在栈上创建对象

只能在栈上创建对象和在堆上类似。 都是将构造函数封掉,然后在类里面创建对象返回给外面。

	//只能在栈上创建对象
	class Only_Stack 
	{
	public:
		//公有一个在栈区创建对象的函数, 并进行值返回
		static Only_Stack* CreatObj()
		{
			Only_Stack obj;
			return &obj;
		}

		//对于new来说。 一般情况下调用自动生成的new。 但是我们也可以自己实现一个operator new。同时, 如果我们禁掉这个new。 那么编译器也不会自动生成了, 参数是size_t
		//void* operator new(size_t size) 
		//{
		//	cout << "operator new" << endl;
		//	return malloc(size);
		//}
	/*	Only_Stack(Only_Stack&) = delete;*/
		void* operator new (size_t size) = delete;

	private:
		//首先私有构造函数。防止能够随便创建对象, 但是为了能在栈区创建对象, 还要定义构造
		Only_Stack() 
		{

		}
	};

不能够继承的类

一个类如果不想被其他的类继承, 那么我们可以给这个类加上关键字final:

	class _final_class final 
	{
		//…………
	};

不能被拷贝的类

一个类如果不能进行拷贝工作, 那么就要将它的拷贝构造和拷贝赋值都封掉:

	//设计一个类, 不能够进行拷贝
	class None_Copy 
	{
	private:
		//将拷贝构造和赋值重载只声明。 不实现。	禁止拷贝, 只需要让这个类无法调用拷贝构造和赋值重载即可。
		//1、为什么要设置成为私有, 是因为如果只声明, 不设置成私有, 那么如果用户在类外面定义了拷贝构造或者赋值重载, 那么就可以进行拷贝或者赋值操作了
		//2、为什么要只声明, 一方面是因为设置成私有并不会禁止类内部进行调用拷贝。 不定义就可以防止类的内部进行拷贝。 另一方面是因为定义了也不会用, 不写更简单
		None_Copy(const None_Copy& cp);
		None_Copy& operator=(const None_Copy& cp);  
	};

以上, 就是本节的全部内容。 有关设计模式的知识, 博主现在学习的不多, 以后学习更多知识后会更新。 当下如果想要学习更多设计模式可以看一下这个大佬(下面放链接了)的文章, 他为自己的文章建立了一个索引:

史上最全设计模式导学目录(完整版)_史上最全设计模式lovelion-CSDN博客

LoveLion-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1834619.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AI绘画Stable Diffusion 保姆级教程,让AI人物轻松打光的种方法,我不允许你还不会!

大家好&#xff0c;我是画画的小强 我们常常听到这样的观点&#xff1a;光影&#xff0c;如同摄影的精髓&#xff0c;为图像赋予难以抗拒的质感和情感深度。 用AI生成的图片为什么总是觉得比较“假”&#xff0c;主要还是光影不足&#xff01; 今天我将一一解析多种光影调控…

MyBatis——增删改查

核心配置文件 MyBatis核心配置文件的顶层结构如下&#xff1a; <environments> 元素 <typeAliases>元素 作用&#xff1a; 配置文件完成增删改查 准备环境 创建数据库表tb_brand -- 删除tb_brand表 drop table if exists tb_brand; -- 创建tb_brand表 create t…

龙芯的 新世界 与 旧世界

但是基本可以 确定 旧世界应该是 有 mips 的代码的。 新世界 应该是 loongarch . 这是 龙芯派 2k300 的连接。 6.Github相关仓库 龙芯派相关源码仓库&#xff1a;https://github.com/LoongsonDotNETCommunity/LoongsonPI 龙芯派Cookbook仓库&#xff1a;https://github.com/L…

UML详解

1.what is the UML UML 全称是 Unified Modeling Language&#xff08;统一建模语言&#xff09;&#xff0c;它以图形的方式来描述软件的概念 2.它存在的目的 UML 的目标是通过一定结构的表达&#xff0c;来解决现实世界到软件世界的沟通问题。 3.什么是模&#xff0c;…

C++语法15 多分支结构(if多分支与switch结构)

if else if else 多分支结构基本框架 if&#xff08;条件1&#xff09;语句1; //满足条件1就执行 else if&#xff08;条件2&#xff09;语句2; //不满足条件1&#xff0c;但是满足条件2执行 else if (条件3&#xff09;语句3; //不满足条件1和条件2&#x…

【链表经典面试题】LeetCode138.复制带随机指针的链表(链表深拷贝)

&#x1f4c7;文章目录 &#x1f680;题目描述&#x1f680;思路1&#xff1a;&#x1f680;思路2&#xff1a;&#x1f680;完整代码 &#x1f680;题目描述 解读&#xff1a; 题目意思就是 给你一个链表 这个链表中除了有next指针之外 还有一个指向这个链表的随机位置的一个指…

在哪里可以查到一手的标讯信息?

标讯信息集招投标讯息的简称。在市场上&#xff0c;标讯是一种非常关键的信息&#xff0c;包括招标公告&#xff0c;文件&#xff0c;截止日期等关键内容&#xff0c;便于需求方和供应商进行业务合作。 对于企业来说&#xff0c;及时获取到最新的标讯信息是非常重要的&#xf…

安装VS Code 提示This User Installer is not meant to be run as an Administrator问题

目录预览 一、问题描述二、原因分析三、解决方案四、参考链接 一、问题描述 在vs code官网&#xff08;https://code.visualstudio.com/&#xff09;下载安装包&#xff0c;显示如下提示信息&#xff1a; This User Installer is not meant to be run as an Administrator.…

【vue大作业-端午节主题网站】【预览展示视频和详细文档】

vue大作业-端午节主题网站介绍 端午节&#xff0c;又称为龙舟节&#xff0c;是中国的传统节日之一&#xff0c;每年农历五月初五庆祝。这个节日不仅是纪念古代爱国诗人屈原的日子&#xff0c;也是家人团聚、共享美食的时刻。今天&#xff0c;我们非常高兴地分享一个以端午节为…

【字符串解析】IP地址字段解析提取函数接口

在嵌入式业务逻辑中&#xff0c;我们有时需要从配置文件、串口或者服务端接收的消息数据中进行字符串解析&#xff0c;来提取需要的目标字符串字段。通常我们会调用字符串处理相关的函数&#xff0c;例如strstr&#xff0c;strchr&#xff0c;sscanf等&#xff0c;再结合指针偏…

【验证码识别】Yolov8实战某验3空间推理点选验证码,目标检测,语义分割,颜色分类。

【验证码识别】Yolov8实战某验3空间推理点选验证码&#xff0c;目标检测&#xff0c;语义分割&#xff0c;颜色分类。 文章目录 【验证码识别】Yolov8实战某验3空间推理点选验证码&#xff0c;目标检测&#xff0c;语义分割&#xff0c;颜色分类。声明1.空间推理验证码&#xf…

C# OpenCvSharp 车牌颜色识别

C# OpenCvSharp 车牌颜色识别 目录 效果 项目 代码 下载 效果 项目 代码 using OpenCvSharp; using System; using System.Diagnostics; using System.Drawing; using System.Windows.Forms; namespace OpenCvSharp_Demo { public partial class Form1 : Form { …

采购芯片时细心,再细心!

检查原理图&#xff0c;采购时候的细心对照所费的时远远少于焊完找BUG的时间&#xff01;&#xff01;&#xff01; 购买芯片的时候不光看芯片名称&#xff0c;封装&#xff0c;丝印也要看&#xff0c;如果不一样必须对照两者的引脚图仔细观察是否一样&#xff01;&#xff01…

扭蛋机小程序:深度探索虚拟寻宝之旅的乐趣

引言 扭蛋机小程序&#xff0c;这个融合了传统与创新的虚拟寻宝乐园&#xff0c;已经吸引了无数玩家的目光。在这个充满惊喜和挑战的虚拟世界里&#xff0c;每一个扭蛋都可能蕴藏着无尽的宝藏。本文将带您深入探索扭蛋机小程序的魅力所在&#xff0c;体验一场别开生面的虚拟寻…

上位机图像处理和嵌入式模块部署(h750 mcu vs f407)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 在目前工业控制上面&#xff0c;f103和f407是用的最多的两种stm32 mcu。前者频率低一点&#xff0c;功能少一点&#xff0c;一般用在低端的嵌入式设…

【数据结构与算法】运算受限的线性表(栈,队列)重要知识点详解

栈和队列是什么样的线性表? 栈&#xff08;Stack&#xff09;和队列&#xff08;Queue&#xff09;都是运算受限的线性表。 栈&#xff1a;栈是一种特殊的线性表&#xff0c;只允许在一端&#xff08;通常称为“顶端”&#xff09;进行插入和删除操作。栈遵循后进先出&#x…

CENTOS7.9下服务器双网卡bond模式6配置示例

​1.bond口的特点 bond口通过将多个网口进行聚合&#xff0c;多个网口聚合后一方面实现了大带宽传输&#xff0c;另外多网口聚合后也同时具有冗余特性&#xff0c;当其中一个网口down掉后&#xff0c;其他网口会继续转发流量&#xff0c;不会导致流量中断。 2.使用条件 当环境…

参数搜索流形学习

目录 一、网格搜索1、介绍2、代码示例 二、HalvingGridSearch1、介绍2、代码示例 三、随机搜索1、介绍2、代码示例 三、贝叶斯搜索1、介绍2、代码示例 四、参数搜索总结五、流形学习1、LLE1、介绍2、官方代码示例 2、t-SNE1、介绍2、官方代码示例 一、网格搜索 1、介绍 网格搜…

安卓手机最近删除照片如何找回?这些技巧来帮你!

我们时常会在手机上拍摄大量照片&#xff0c;记录下生活中的每一个瞬间。然而&#xff0c;由于存储空间不足、设备更新等原因&#xff0c;我们可能会不小心删除一些照片。最近删除照片如何找回&#xff1f;通过本文的介绍&#xff0c;您将了解到如何轻松找回最近删除的照片&…

2025计算机毕业设计选题题目推荐-毕设题目汇总大全

选题在于精&#xff0c;以下是推荐的容易答辩的选题&#xff1a; SpringBoot Vue选题: 基于SpringBoot Vue家政服务系统 基于SpringBoot Vue非物质文化遗产数字化传承 基于SpringBoot Vue兽医站管理系统 基于SpringBoot Vue毕业设计选题管理系统 基于SpringBoot Vue灾害应急救援…