C++中特殊类的设计与单例模式的简易实现

设计一个只能在堆上创建对象的类

对于这种特殊类的设计我们一般都是优先考虑私有构造函数。然后对于一些特殊要求就直接通过静态成员函数的实现来完成。

class A//构造函数私有(也可以析构函数私有)
{
public:static A* creat(){return new A;}
private:A(){}A(const A&) = delete;A operator=(const A&) = delete;int _a;
};

 这里选择禁掉拷贝构造函数和拷贝函数是为了防止将已创建的对象去拷贝构造新的对象。

设计一个只能在栈上创建对象的类

class B
{
public:static B creat(){B tmp;return tmp;}//直接禁掉new和delete//全局也有,类中也有,会优先调用类中的operator new(类中专属的)void* operator new(size_t size) = delete;void operator delete(void* p) = delete;
private:B(){}int _b;
};

 这里如果没有禁掉operator new和operator delete的话就会导致以下情况是在栈上创建对象

B* pb = new B(B::creat());//会调用拷贝构造

对于这种new一个对象的情况下是会调用拷贝构造函数的 ,而且我们是不能直接禁掉拷贝构造函数的,因为我们创建的对象必须通过调用拷贝构造函数来接受,所以就有了禁掉operator new和operator delete的方式。

对于在类中禁掉这两个函数,我们需要了解:当我们类中实现了operator new和operator delete这两个函数的话,在我们new该类对象或delete该类对象的话就会调用该类的operator new和operator delete函数,而不会选择调用全局的operator new和operator delete函数。

设计一个不能被继承的类

class D final
{D()//私有构造也可以{}
};

这里需要认识一个关键词final,该关键词修饰得类会不允许被继承。 

而且还有一点,我们知道继承一个类之后,子类创建对象会调用父类的构造函数,如果父类没有默认的构造函数的话,子类必须要显示的调用父类的构造函数。

设计一个只能创建一个对象的类(单例模式)

单例模式:一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。
单例模式有以下两种实现方式:

饿汉模式

饿汉模式就是不管你用不用该实例,系统在启动程序时就会直接创建一个唯一的实例。

class A
{A(const A& tmp) = delete;//禁掉拷贝构造,防止再次创建对象A& operator=(const A& tmp) = delete;public:static A& getinstance()//通过成员函数得到该对象{return sigle;}void Add(string s1, string s2){_dict[s1] = s2;}void Print(){for (auto tmp : _dict){cout << tmp.first << ":" << tmp.second << endl;}}private:A()//只能创建一个对象,构造私有{cout << "构造完成" << endl;}map<string, string> _dict;static A sigle;//声明 类里面静态成员可以直接调用私有函数
};
A A::sigle;//定义(此时已经调用好了构造函数)

对于单例模式一般就是在类中提前声明好该静态对象 所以在定义自定义类型的的时候就会直接调用构造函数。而这sigle对象就是我们所创建的唯一实例。 

懒汉模式

懒汉模式就是在你开始调用的时候才会创建对象。

class B
{B(const B& tmp) = delete;//禁掉拷贝构造,防止再次创建对象B& operator=(const B& tmp) = delete;
public:static B* getinstance(){if (sigle == nullptr)//只有为空才创建sigle = new B;//一般不需要释放,进程结束的时候会释放return sigle;}static void del(){delete sigle;//为空就不会再调用析构函数sigle = nullptr;}void Add(string s1, string s2){_dict[s1] = s2;}void Print(){for (auto tmp : _dict){cout << tmp.first << ":" << tmp.second << endl;}}
private:B()//只能创建一个对象,构造私有{cout << "构造完成" << endl;}~B(){//持久化:要求数据写到文件中cout << "数据录入文件中 并析构" << endl;}map<string, string> _dict;static B* sigle;//内部类是外部类的友元class gc//类似智能指针,程序结束前调用析构{public:gc(){cout << "gc()" << endl;}~gc(){del();}};static gc _gc;//声明  创建静态成员,属于一个类,不会创建多份//程序结束前就会自动调用gc析构函数};
B* B::sigle = nullptr;//初始化为空
B::gc B::_gc;//定义  main结束会调用构造

对于懒汉模式和饿汉模式本质区别就是对象和指针的转变,饿汉模式的唯一实例是一个静态的类指针,但是该指针的释放就会有点困难,需要我们手动去delete,其实进程结束也是会释放内存的,但是对于一些需要持久化的将数据写到文件的情况时就会采用内部类gc来解决(内部类是外部类的友元,可以访问外部类的所有成员)而同样在外部类中创建静态成员,类外进行定义(调用构造),所以当main函数结束前,对于自定义的该成员就会自动调用析构函数,此时就可以进行持久化处理。

对于以上代码如果_gc不设为外部类的静态成员而是一般成员的话,就会陷入循环析构,因为对于外部类来说_gc属于类的自定义成员,所以当调用外部类的析构时就会先调用自生的析构函数再调用自定义成员_gc的析构函数,此时就会造成循环析构的情况。

int main()
{B::getinstance()->Add("sort", "排序");B::getinstance()->Add("left", "左边");B::getinstance()->Print();//B::del();return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Redis】基于Token单点登录

基于Token单点登录 获取验证码 流程图 #mermaid-svg-DLGHgCofEYXVSmI5 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-DLGHgCofEYXVSmI5 .error-icon{fill:#552222;}#mermaid-svg-DLGHgCofEYXVSmI5 .error-text{f…

Oracle行转列函数,列转行函数

Oracle行转列函数&#xff0c;列转行函数 Oracle 可以通过PIVOT,UNPIVOT,分解一行里面的值为多个列,及来合并多个列为一行。 PIVOT PIVOT是用于将行数据转换为列数据的查询操作(类似数据透视表)。通过使用PIVOT&#xff0c;您可以按照特定的列值将数据进行汇总&#xff0c;并将…

【webrtc】GCC 7: call模块创建的ReceiveSideCongestionController

webrtc 代码学习&#xff08;三十二&#xff09; video RTT 作用笔记 从call模块说起 call模块创建的时候&#xff0c;会创建 src\call\call.h 线程&#xff1a; 统计 const std::unique_ptr<CallStats> call_stats_;SendDelayStats &#xff1a; 发送延迟统计 const…

用Go plan9汇编实现斐波那契数列计算

斐波那契数列是一个满足递推关系的数列&#xff0c;如&#xff1a;1 1 2 3 5 8 ... 其前两项为1&#xff0c;第3项开始&#xff0c;每一项都是其前两项之和。 用Go实现一个简单的斐波那契计算逻辑 func fib(n int) int {if n 1 || n 2 {return 1}return fib(n-1) fib(n-2) …

RT-Thread experimental 代码学习(1)thread_sample

RTOS的最基础功能是线程。 线程的调度是如何工作的&#xff1f;RT-thread官方的实验文档是最好的参考。 老规矩&#xff0c;先放法国人doxygen。 thread_sample 代码的调用关系图 有意思的是&#xff0c;RT有两种创建线程的方式 - 静态和动态&#xff0c;粗略的理解是&…

Qt弹框展示

1.相关说明 文件选择弹框、目录选择弹框、保存文件弹框、颜色选择弹框、字体选择弹框、进度条弹框、输入对话框、标准消息框等 2.相关界面 3.相关代码 #include "widget.h" #include "ui_widget.h" #include <QFileDialog> #include <QProgressD…

Windows下安装alipay-sdk-python时,pycrypto安装报错问题处理

1、安装alipay-sdk-python 时&#xff0c;保存内容如下。 Building wheels for collected packages: pycryptoBuilding wheel for pycrypto (setup.py) ... error error: subprocess-exited-with-error python setup.py bdist_wheel did not run successfully.│ exit c…

Conway的生命游戏

文章目录 主要内容一.Conway的“生命游戏”1.玩法代码如下&#xff08;示例&#xff09;: 总结 主要内容 一.Conway的“生命游戏” 1.玩法 Conway的“生命游戏”是细胞自动机的一个例子&#xff1a;一组规则控制由离散细胞组成的区域的行为。在实践中&#xff0c;它会创建一个…

【办公类-21-02】20240118育婴员操作题word打印2.0

作品展示 把12页一套的操作题批量制作10份&#xff0c;便于打印 背景需求 将昨天整理的育婴师操作题共享&#xff0c; 因为题目里面有大量的红蓝颜色文字&#xff0c;中大班办公室都是黑白单面手动翻页打印。只有我待的教务室办公室有彩色打印机打印&#xff08;可以自动双面…

Gitlab添加ssh-key报500错误处理

Gitlab添加ssh-key报500错误 一、查看日志 发现Errno::Enoent(No such file or derectory -ssh): rootasu1:/home/caixin# tail -f /var/log/gitlab/gitlab-rails/production.log二、分析 根据日志提示&#xff0c;好像是缺少文件或目录&#xff0c;后面有个ssh,难首是依赖s…

CVE重要通用漏洞复现java php

在进行漏洞复现之前我们需要在linux虚拟机上进行docker的安装 我不喜欢win上安因为不知道为什么总是和我的vmware冲突 然后我的kali内核版本太低 我需要重新安装一个新的linux 并且配置网络 我相信这会话费我不少时间 查看版本 uname -a 需要5.5或以上的版本 看错了浪…

Leetcode2182. 构造限制重复的字符串

Every day a Leetcode 题目来源&#xff1a;2182. 构造限制重复的字符串 解法1&#xff1a;贪心 双指针 我们先用一个长度为 26 的数组 cnt 统计字符串 s 中每个字符出现的次数&#xff0c;然后从大到小枚举字母表的第 i 个字母&#xff0c;每次取出最多 min⁡(cnt[i], re…