【小梦C嘎嘎——启航篇】C++特殊类设计

2024-04-28 23:36

本文主要是介绍【小梦C嘎嘎——启航篇】C++特殊类设计,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【小梦C嘎嘎——启航篇】C++特殊类设计😎

  • 前言🙌
    • 1.请设计一个类,该类不能被继承
    • 2.请设计一个类,只能在堆上创建对象
    • 3.请设计一个类,只能在栈上创建对象
    • 4.请设计一个类,该类不能发生拷贝
    • 5.请设计一个类,该类只能创建一个对象
      • 什么是单例模式
      • 饿汉模式
      • 懒汉模式的实现
  • 总结撒花💞

追梦之旅,你我同行

   
😎博客昵称:博客小梦
😊最喜欢的座右铭:全神贯注的上吧!!!
😊作者简介:一名热爱C/C++,算法等技术、喜爱运动、热爱K歌、敢于追梦的小博主!

😘博主小留言:哈喽!😄各位CSDN的uu们,我是你的博客好友小梦,希望我的文章可以给您带来一定的帮助,话不多说,文章推上!欢迎大家在评论区唠嗑指正,觉得好的话别忘了一键三连哦!😘
在这里插入图片描述

前言🙌

    哈喽各位友友们😊,我今天又学到了很多有趣的知识现在迫不及待的想和大家分享一下!😘都是精华内容,可不要错过哟!!!😍😍😍

1.请设计一个类,该类不能被继承

实现思想:

  • 将构造函数和拷贝构造函数私有化
  • 用C++11关键字final修饰类,表示这个类不能被继承
class B {
private:B() {};B(const B& b) = delete;
};class B final{
};class C :public B{
public:C(){}
};int main()
{C c1;
}

2.请设计一个类,只能在堆上创建对象

实现思路:
方案一:构造函数私有化,防拷贝。通过公共接口创建堆上的对象

class HeapOnly
{
public:static HeapOnly* CreateObj(){return new HeapOnly;}HeapOnly(const HeapOnly& hp) = delete;
private:HeapOnly(){cout << "HeapOnly()" << endl;}
};

方案二:析构函数私有化:由于h1对象释放时,会自动调用析构,但是析构已经被私有化了,因此会编译错误。

class HeapOnly
{
public:void Destroy(){delete this;}
private:~HeapOnly(){cout << "~HeapOnly()" << endl;}
};
int main()
{//HeapOnly h1;HeapOnly *h1 = new HeapOnly();h1->Destroy();return 0;
}

3.请设计一个类,只能在栈上创建对象

实现思路:将构造函数私有化,把operator new禁掉。一个类可以设置专属的,就不会去调用全局的operator new。

class StackOnly
{
public:static StackOnly CreateObj(){StackOnly obj1;return obj1;}//将new禁掉void* operator new(size_t size) = delete;private:// 构造函数私有化StackOnly(){cout << "HeapOnly()" << endl;}
};int main()
{StackOnly s1 = StackOnly::CreateObj();return 0;
}

4.请设计一个类,该类不能发生拷贝

实现思路:防拷贝,防赋值

//c++ 98实现
class A {private:A(const A& a1);A& operator=(const A& a1);
};
//c++11 实现
class A {A(const A& a1) = delete;A& operator=(const A& a1) = delete;
};

5.请设计一个类,该类只能创建一个对象

这里的设计模式有23种,这里只实现单例模式
单例模式又分为两种实现模式:饿汉模式和懒汉模式。这里实现没有加入线程的知识,因为博主还没有学到线程的知识,所以暂时先简单的设计。后面学习了线程后,在考虑线程安全问题,对这部分的实现进行一个完善。

什么是单例模式

一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。如何是设计只有一个内存池的场景时,可以使用单例模式进行一个设计。

饿汉模式

饿汉模式:就是说不管你将来用不用,程序启动时就创建一个唯一的实例对象**。也就是说在main函数启动时,提前创建好实例对象**

  • 优点:实现简单
  • 缺点:1、可能会导致进程启动慢、2、如果两个单例有启动先后顺序,那么饿汉无法控制

实现思路:

  • 首先还是先把构造函数私有化、拷贝构造函数私有化,最好把赋值重载函数也一样私有化或者禁掉。

  • 要想只生成一个共有的对象,并且该对象是在mian函数启动前创建好,可以在类内定义一个static singletion m_instance 成员变量。它是属于整个类的,而不是每一个对象的。确保只有这一个对象。

有的小伙伴可能会想到定义成全局的对象,但是会存在一些缺点:

  1. 构造函数私有,创建不出来全局对象
  2. 全局对象容易被人修改
  3. 如果在一个.h文件中定义全局变量,有多个.cpp文件包含了这个头文件,链接时会出现重定义问题。
class singletion {
public://被访问的公共节点static  singletion* GetInstance(){return &m_instance;}int& get_a(){return _a;}void Print(){cout << _a << endl;}
private:singletion() = delete;singletion& operator=(const singletion& s) = delete;singletion(const singletion& s) = delete;static singletion m_instance;int _a = 0;
};
//初始化
singletion singletion:: m_instance

懒汉模式的实现

懒汉模式:第一次使用实例对象时,创建对象(现吃现做)

缺点:实现复杂
优点:第一次使用实例对象时,创建对象。进程启动无负载。多个单例实例启动顺序自由控制。

实现思路:
1.构造函数私有化,防拷贝,防赋值
2.设计一个静态的对象指针成员变量,保证这个指针是这个类的,确保它是单例的。后面通过公共接口调用该指针申请对象。实现只有第一次调用时动态申请对象实例。
3.这里实现了手动释放对象的空间资源和main函数结束后,自动释放两种回收资源的方式。实现自动回收,则在B类中定义一个内部类,通过gc对象结束生命周期,自动调用它的析构,然后在调用B类的析构函数,从而完成资源的释放。


class B
{
public:static B* GetInstance(){if (_inst == nullptr){_inst = new B;}return _inst;}//手动释放---直接外面调用该接口static void DelInstance(){if (_inst){delete _inst;_inst = nullptr;}}private:B(){}~B(){// 持久化:要求把数据写到文件cout << "数据写到文件" << endl;}B(const B& aa) = delete;B& operator=(const B& aa) = delete;int _n = 0;static B* _inst;class gc{public://通过内部类的析构函数调用B类的析构接口~gc(){DelInstance();}};static gc _gc;
};B* B::_inst = nullptr;
//全局只有一个gc对象,刚好对应释放一次B对象的资源空间,
//当main函数结束,gc也会结束,从而自动回收单例对象的资源
B::gc B::_gc;

一般来说,new的懒汉对象不需要提前释放,进程正常结束会释放资源。这里是操作系统层次上的操作,不会调用析构函数。如果对于有些场景,析构函数内部需要完成将数据拷贝回硬盘再析构的话,这里就会出现问题。因此,可以设计出一个内部类,通过内部类对象结束访问到其析构函数,从而完成拷贝并回收资源的工作。

总结撒花💞

   本篇文章旨在分享的是C++特殊类设计知识。希望大家通过阅读此文有所收获
   😘如果我写的有什么不好之处,请在文章下方给出你宝贵的意见😊。如果觉得我写的好的话请点个赞赞和关注哦~😘😘😘

这篇关于【小梦C嘎嘎——启航篇】C++特殊类设计的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/944561

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

C/C++的编译和链接过程

目录 从源文件生成可执行文件(书中第2章) 1.Preprocessing预处理——预处理器cpp 2.Compilation编译——编译器cll ps:vs中优化选项设置 3.Assembly汇编——汇编器as ps:vs中汇编输出文件设置 4.Linking链接——链接器ld 符号 模块,库 链接过程——链接器 链接过程 1.简单链接的例子 2.链接过程 3.地址和

C++必修:模版的入门到实践

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C++学习 贝蒂的主页:Betty’s blog 1. 泛型编程 首先让我们来思考一个问题,如何实现一个交换函数? void swap(int& x, int& y){int tmp = x;x = y;y = tmp;} 相信大家很快就能写出上面这段代码,但是如果要求这个交换函数支持字符型

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

在线装修管理系统的设计

管理员账户功能包括:系统首页,个人中心,管理员管理,装修队管理,用户管理,装修管理,基础数据管理,论坛管理 前台账户功能包括:系统首页,个人中心,公告信息,论坛,装修,装修队 开发系统:Windows 架构模式:B/S JDK版本:Java JDK1.8 开发工具:IDEA(推荐) 数据库版本: mysql5.7 数据库可视化工具: navicat 服务器:SpringBoot自带 ap

C++入门01

1、.h和.cpp 源文件 (.cpp)源文件是C++程序的实际实现代码文件,其中包含了具体的函数和类的定义、实现以及其他相关的代码。主要特点如下:实现代码: 源文件中包含了函数、类的具体实现代码,用于实现程序的功能。编译单元: 源文件通常是一个编译单元,即单独编译的基本单位。每个源文件都会经过编译器的处理,生成对应的目标文件。包含头文件: 源文件可以通过#include指令引入头文件,以使

DDei在线设计器-API-DDeiSheet

DDeiSheet   DDeiSheet是代表一个页签,一个页签含有一个DDeiStage用于显示图形。   DDeiSheet实例包含了一个页签的所有数据,在获取后可以通过它访问其他内容。DDeiFile中的sheets属性记录了当前文件的页签列表。   一个DDeiFile实例至少包含一个DDeiSheet实例。   本篇最后提供的示例可以在DDei文档直接预览 属性 属性名说明数