【C++高阶(八)】单例模式特殊类的设计

💓博主CSDN主页:杭电码农-NEO💓

⏩专栏分类:C++从入门到精通⏪

🚚代码仓库:NEO的学习日记🚚

🌹关注我🫵带你学习C++
  🔝🔝


在这里插入图片描述

单例模式

  • 1. 前言
  • 2. 设计一个不能被拷贝/继承的类
  • 3. 只能在堆上创建对象的类
  • 4. 只能在栈上创建对象的类
  • 5. 只能实例化一个对象的类的介绍
  • 6. 饿汉模式的具体实现
  • 7. 懒汉模式的具体实现
  • 8. 总结以及拓展

1. 前言

在实际场景中,总会遇见一些特殊情况,
比如设计一个类,只能在堆上开辟空间,
亦或者是设计一个类只能实例化一个对象
在实际需求的场景下,来学习这节实用课

本章重点:

本篇文章着重讲解如何设计一些特殊
的类,包括不能被拷贝,只能在栈/堆上
创建对象以及此类只能实例化一个对象,
这也就是题目中的单例模式,单例模式又
包含饿汉和懒汉模式,文章都是干货
请同学们耐心学习!


2. 设计一个不能被拷贝/继承的类

  1. 设计一个不能被拷贝的类

C++11中引入的关键字delete
就能很好的解决这个问题,并且
不仅仅要禁用拷贝,还有赋值!

class CopyBan
{
    CopyBan(const CopyBan&)=delete;
    CopyBan& operator=(const CopyBan&)=delete;
};

在C+98中,也有方法能够解决,
那就是显示将拷贝构造函数和
赋值运算符重载函数私有化!

class CopyBan
{
private:
    CopyBan(const CopyBan&);
    CopyBan& operator=(const CopyBan&);
};
  1. 设计一个不能被继承的类

使用关键字final就能解决问题

class A  final
{
    // ....
};

在C++98中,将构造函数私有化也能
达到目的,因为子类的构造会调用基类
的构造,如果私有了基类的构造就会报错!

class NonInherit
{
private:
	NonInherit()
	{}
};

3. 只能在堆上创建对象的类

只能在堆上创建对象的含义就是
必须使用new来创建对象.

本篇文章是实用性的,就直接讲方法了:

  1. 将析构函数私有化

将析构函数私有化后,由于对象析构时并不能调用到析构函数,所以不管是在堆上还是栈上创建对象都会报错!但是我们可以特殊处理,在共有域定义一个函数,此函数显示调用析构!

//思路一,封析构函数
class HeapOnly
{
public:
	void destory()
	{
		delete this;
	}
private:
	~HeapOnly()
	{
		cout<<"调用析构成功!"<<endl;
	}
};

能否达到目的大家可以自行测试!

  1. 将构造函数私有化

将构造函数设置为私有后,不管是在堆上还是栈上都不能创建对象,但是我们可以在共有域写一个函数显示去调用构造函数,注意,这里的共有域函数必须设置为static类型,因为必须有了对象后才能调用函数,但是要调用了此函数才能创建对象,就会出现先有鸡还是先有蛋的问题,所以设置为static后,可以用类域调用!

//思路二,封构造函数
class HeapOnly
{
public:
	static HeapOnly* CreateObject(int x = 0)
	{
		return new HeapOnly(x);
	}
private:
	HeapOnly(int x = 0):_x(x)
	{}
	int _x;
};
  1. 以上两种方案真的就ok了吗?

事实上并不够ok,因为即使封掉了析构
或者是构造,人们也能用拷贝构造或
赋值来在栈上开辟空间,比如在方法二
中,我们可以这样打破规则:

HeapOnly* ho1 = HeapOnly::CreateObject(10);
HeapOnly ho(*ho1);

所以在上面两种方案的基础上
要禁用拷贝构造和赋值重载两个函数!


4. 只能在栈上创建对象的类

有了前面的思想,解决这个类型
的问题就显示很小儿科了!

同上将构造函数私有化然后设计
静态方法创建对象返回对象即可

class StackOnly
{
public:
	static StackOnly CreateObj()
	{
		return StackOnly();
	}

// 禁掉operator new可以把下面用new 调用拷贝构造申请对象给禁掉
// StackOnly obj = StackOnly::CreateObj();
// StackOnly* ptr3 = new StackOnly(obj);
	void* operator new(size_t size) = delete;
	void operator delete(void* p) = delete;
private:
	StackOnly()  
		:_a(0)
	{}
	private:
	int _a;
};

5. 只能实例化一个对象的类的介绍

一个类只能实例化一个对象
这就是大名鼎鼎的"单例模式"

谈单例模式前,先谈设计模式:

在这里插入图片描述

单例模式就是设计模式中的一种:

在这里插入图片描述

单例模式在实际场景下使用非常广泛
如果你恰好在读我的并发内存池项目
亦或者是你学过线程池(thread pool),
这里都能看见单例模式的影子,并且,
单例模式有两种实现模式:

  • 饿汉模式:就是说不管你将来用不用,程序启动时就创建一个唯一的实例对象
  • 懒汉模式:第一次使用时才创建一个唯一的实例对象

6. 饿汉模式的具体实现

注意,这里实现的是样例(demo)代码,在
不同的工程场景下需要大家做灵活的变换

// 饿汉模式
// 优点:简单
// 缺点:可能会导致进程启动慢,且如果有多个单例类对象实例启动顺序不确定。
class Singleton
{
public:
	static Singleton* GetInstance()
	{
		return _ins;
	}
private:
	//限制类外随意创建对象
	Singleton(const Singleton& s) = delete;
	Singleton& operator=(const Singleton& s) = delete;
	Singleton()
	{}
private:
	static Singleton* _ins;
};
Singleton* Singleton::_ins = new Singleton;

单例模式的饿汉模式中,程序一启动就会
把_ins,也就是唯一的实例对象给初始化,
并且由于构造函数被私有了,只能调用共
有的GetInstance()函数获取_ins对象,又
由于这个对象是static类型的,所以不管你
调用多少次GetInstance()都获取的是同
一个对象,也就是_ins


7. 懒汉模式的具体实现

在这里插入图片描述

//懒汉模式
class Singleton
{
public:
	static Singleton* GetInstance()
	{
		if (_ins == nullptr)//双检查加锁,只有第一次进来时需要加锁,其他情况不用加锁
		{
			imtx.lock();
			if (_ins == nullptr)//第一次调用才创建实例!
			{
				_ins = new Singleton;
			}
			imtx.unlock();
		}
		
		return _ins;
	}
	void DelInstance()
	{
		imtx.lock();
		if (_ins != nullptr)
		{
			cout << "over!!!" << endl;
			delete _ins;
			_ins = nullptr;
		}
		imtx.unlock();
	}
private:
	//限制类外随意创建对象
	Singleton(const Singleton& s) = delete;
	Singleton& operator=(const Singleton& s) = delete;
	Singleton()
	{}
private:
	static Singleton* _ins;
	static mutex imtx;
};
Singleton* Singleton::_ins = nullptr;
mutex Singleton::imtx;

与饿函数模式不同的是,懒汉模式在多线程
情况下有线程安全问题,所以在第一次拿唯
一的对象前需要加锁,并且对象在程序启动
时被置空了,只有调用了GetInstance()才会
真正的分配空间

当然,这两个模式都是样例代码,大家要随机应变


8. 总结以及拓展

特殊类的设计这块儿,大家需要在写某些
项目的时候真正运用到它才能体会出它
的作用和奥妙之处,总的来说单例模式是
使用很广泛并且很有用的一种设计模式!

对设计模式的拓展:

常见的设计模式不仅仅有单例模式,还有工厂模式、抽象工厂模式、适配器模式、装饰者模式、代理模式、外观模式、桥接模式、组合模式、享元模式、观察者模式和命令模式等,如果大家有兴趣的话可以阅读这篇文章拓展自己的知识

C++常见的11种设计模式


🔎 下期预告:C++类型转换以及IO流🔍

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

外贸业务员该如何写好一份有质感的年终总结?内附外贸大神例文

庄子云&#xff1a;人生天地之间&#xff0c;若白驹之过隙&#xff0c;忽然而已... 2023年注定是不平凡的一年&#xff0c;临近年终&#xff0c;你可能听到最多的关键词就是外贸有点难做。不管是因为什么&#xff0c;客观来说2023年的外贸之路确实不太平坦&#xff0c;最近胡塞…

4. 行为模式 - 中介者模式

亦称&#xff1a; 调解人、控制器、Intermediary、Controller、Mediator 意图 中介者模式是一种行为设计模式&#xff0c; 能让你减少对象之间混乱无序的依赖关系。 该模式会限制对象之间的直接交互&#xff0c; 迫使它们通过一个中介者对象进行合作。 问题 假如你有一个创建…

Catboost算法助力乳腺癌预测:Shap值解析关键预测因素

一、引言 乳腺癌是一种常见的恶性肿瘤&#xff0c;对女性健康和生命造成严重威胁。乳腺癌的预测和治疗是当前研究的热点和难点。传统的预测方法主要基于临床病理学特征&#xff0c;但准确率有待提高。随着机器学习技术的发展&#xff0c;数据驱动的预测方法逐渐受到关注。Catbo…

由浅入深,一文读懂网络知识文集。

&#x1f3c6;作者简介&#xff0c;普修罗双战士&#xff0c;一直追求不断学习和成长&#xff0c;在技术的道路上持续探索和实践。 &#x1f3c6;多年互联网行业从业经验&#xff0c;历任核心研发工程师&#xff0c;项目技术负责人。 &#x1f389;欢迎 &#x1f44d;点赞✍评论…

【SpringMVC】SpringMVC的请求与响应

文章目录 0. Tomcat环境的配置1. PostMan工具介绍创建WorkSpace建立新的请求 2. 请求映射路径案例结构与代码案例结构案例代码 案例存在问题解决方案方法方法升级版——配置请求路径前缀注解总结 3. Get请求与Post请求案例结构与案例代码案例结构案例代码 Get请求Post请求接收中…

熔断,降级,限流

文章目录 熔断概念为什么需要熔断熔断器模型熔断策略 降级概念熔断和降级的关系熔断降级限时降级限流降级 参考文章 熔断和降级都是自我保护的一种机制&#xff0c;但二者又有所不同。 熔断 概念 “熔断”一词早期来自股票市场。熔断[Circuit Breaker]也叫自动停盘机制&#…

Gradle 简单入门

Gradle简单介绍&#xff1a; Gradle是一个基于Apache Ant和Apache Maven概念的项目自动化构建开源工具。它使用一种基于Groovy的特定领域语言(DSL)来声明项目设置&#xff0c;也增加了基于Kotlin语言的kotlin-based DSL&#xff0c;抛弃了基于XML的各种繁琐配置。面向Java应用…

【GoLang】哪些大公司正在使用Go语言

你见过哪些令你膛目结舌的代码技巧&#xff1f; 文章目录 你见过哪些令你膛目结舌的代码技巧&#xff1f;前言&#xff1a;哪些大公司正在使用Go语言谷歌&#xff08;Google&#xff09;&#xff1a;脸书&#xff08;Facebook&#xff09;&#xff1a;亚马逊&#xff08;Amazon…

TikTok手工艺新潮流:小视频中的手作乐趣

随着短视频平台的兴起&#xff0c;TikTok以其独特的创意和活力&#xff0c;成为了全球用户分享生活、展示才华的热门平台之一。本文将深入探讨TikTok上涌现的手工艺潮流&#xff0c;揭示小视频中的手作乐趣如何成为一种全新的艺术表达方式。 TikTok手工艺的独特之处 TikTok提供…

实验一传统的结构化的软件工程方法、实验二面向对象的软件工程、实验三软件测试

背景&#xff1a; 实验一 传统的结构化的软件工程方法 1实验目的 了解传统的软件工程方法的基本原理&#xff0c;掌握软件生命周期的全过程依次划分为需求分析、总体设计、详细设计、编码、测试、维护等几个重要阶段。每个阶段所要完成的任务以及提交的文档。 2实验内容 …

32 在Vue3中如何同时定义多个插槽

概述 当你想要给外部预留多个位置的时候&#xff0c;具名插槽就非常有用了。 比如&#xff0c;我们定义一个卡片&#xff0c;让别人使用的时候&#xff0c;标题可以自定义&#xff0c;内容也可以自定义&#xff0c;这个时候就需要两个插槽。 基本用法 我们创建src/componen…

行业前景咋样?大厂找我用C++抓取化工产品数据并分析

最近又来活了&#xff0c;天天忙到半夜&#xff0c;但是想想收益还是再坚持坚持。是这么一回事&#xff0c;兄弟所在的化工公司最近想看看某些行业数据&#xff0c;看看市面的同类型产品销量收益等情况是否满足预期效果&#xff0c;也就找到我让我给用爬虫写一个采集并分析的报…

基于Java SSM框架实现教学质量评价评教系统项目【项目源码+论文说明】

基于java的SSM框架实现教学质量评价评教系统演示 摘要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;教学质量评价系统当然也不能排除在外。教学质量评价系统是以实际运用为…

机器学习 | 密度聚类和层次聚类

密度聚类和层次聚类 密度聚类 背景知识 如果 S 中任两点的连线内的点都在集合 S 内&#xff0c;那么集合 S称为凸集。反之&#xff0c;为非凸集。 DBSCAN 算法介绍 与划分和层次聚类方法不同&#xff0c;DBSCAN(Density-Based Spatial Clustering of Applications with Noi…

【模式识别】解锁降维奥秘:深度剖析PCA人脸识别技术

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《模式之谜 | 数据奇迹解码》⏰诗赋清音&#xff1a;云生高巅梦远游&#xff0c; 星光点缀碧海愁。 山川深邃情难晤&#xff0c; 剑气凌云志自修。 目录 &#x1f30c;1 初识模式识…

MySQL基本操作 DDL DML DQL三大操作介绍

DDL 数据(结构)定义 创建表DML 数据操作 增删改DQL 查询语句 DDL 数据(结构)定义 创建表 创建 删除数据 注释 --空格内容 创建数据库 CREATE DATABASE [if not exists] 数据库名 [ CHARSET utf8]eg:CREATE DATABASE IF NOT EXISTS school CHARSET utf8如果对应school不存在,…

Python学习笔记(六):函数的多返回值、函数的多种参数使用形式、匿名函数、文件的读取操作、文件的写入 、文件的追加

目录 一、函数的多返回值 二、函数的多种参数使用形式 2.1位置参数 2.2关键字参数 2.3缺省参数 2.4不定长参数 三、匿名函数 3.1 函数作为参数传递 3.2 函数的定义 3.3 匿名函数定义语法&#xff1a; 四、文件的读取操作 4.1 open&#xff08;&#xff09;打开函数…

2023 英特尔On技术创新大会直播 |我感受到的“芯”魅力

文章目录 每日一句正能量前言AI时代&#xff0c;云与PC结合为用户带来更好体验全新处理器&#xff0c;首次引入针对人工智能加速的NPU大模型时代&#xff0c;软硬结合带来更好训练成果后记 每日一句正能量 成长是一条必走的路路上我们伤痛在所难免。 前言 在2023年的英特尔On技…

【UML】第8篇 用例图(3/3)

目录 一、用例的关系 1.1 泛化&#xff08;Generalization&#xff09;关系 1.2 包含&#xff08;include&#xff09;关系 1.3 扩展关系 二、用例表示例 不是非要把电影改成连续剧&#xff0c;给大家播&#xff0c;确实是时间和精力有限。 用例图&#xff0c;虽然简单&…

一个网卡能设置(绑定)两个或多个IP

昨天领导问我&#xff0c;一个网卡设置两个IP。我就有点发愣&#xff1a;一个网卡能设置两个IP吗&#xff1f;我倒是见过一个机器插两个网卡&#xff0c;同时支持内网和外网。一个网卡设置两个IP&#xff0c;以谁为准&#xff1f; 上网搜索了一下&#xff0c;还真可以&#xff…