C++特殊类设计单例模式...

文章目录

    • 请设计一个类,不能被拷贝
    • 请设计一个类,只能在堆上创建对象
    • 请设计一个类,只能在栈上创建对象
    • 请设计一个类,不能被继承
    • 请设计一个类,只能创建一个对象(单例模式)
      • 单例模式:
        • 饿汉模式:
        • 懒汉模式:
      • 懒汉模式的线程安全问题
        • 工厂模式
        • 观察者模式

请设计一个类,不能被拷贝

  • 拷贝只会放生在两个场景中:拷贝构造函数以及赋值运算符重载,因此想要让一个类禁止拷贝, 只需让该类不能调用拷贝构造函数以及赋值运算符重载即可。

C++98:

class CopyBan {// ...private: CopyBan(const CopyBan&); CopyBan& operator=(const CopyBan&);//... 
};

原因:

  1. 设置成私有:如果只声明没有设置成private,用户自己如果在类外定义了,就可以不 能禁止拷贝了
  2. 只声明不定义:不定义是因为该函数根本不会调用,定义了其实也没有什么意义,不写 反而还简单,而且如果定义了就不会防止成员函数内部拷贝了。

C++11:

  • C++11扩展delete的用法,delete除了释放new申请的资源外,如果在默认成员函数后跟上 =delete,表示让编译器删除掉该默认成员函数。
class CopyBan {
CopyBan(const CopyBan&) = delete;
CopyBan& operator=(const CopyBan&) = delete;
};

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

实现方式:

  1. 将类的构造函数私有,拷贝构造声明成私有。防止别人调用拷贝在栈上生成对象。
  2. 提供一个静态的成员函数,在该静态成员函数中完成堆对象的创建
class HeapOnly
{
public:static HeapOnly* CreateObject(){return new HeapOnly;}
private:HeapOnly() {}// C++98// 1.只声明,不实现。因为实现可能会很麻烦,而你本身不需要// 2.声明成私有HeapOnly(const HeapOnly&);// C++11    HeapOnly(const HeapOnly&) = delete;
};

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

  • 方法一:同上将构造函数私有化,然后设计静态方法创建对象返回即可。
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;
};

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

C++98方式:

  • C++98中构造函数私有化,派生类中调不到基类的构造函数。则无法继承
class NonInherit
{
public:static NonInherit GetInstance(){return NonInherit();}
private:NonInherit(){}
};

C++11方法:

  • final关键字,final修饰类,表示该类不能被继承。
class NonInherit final
{
public:// ...
};

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

设计模式:

  • 设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类的、代码设计经验的 总结。为什么会产生设计模式这样的东西呢?就像人类历史发展会产生兵法。最开始部落之间打 仗时都是人拼人的对砍。后来春秋战国时期,七国之间经常打仗,就发现打仗也是有套路的,后 来孙子就总结出了《孙子兵法》。孙子兵法也是类似。

  • 使用设计模式的目的:为了代码可重用性、让代码更容易被他人理解、保证代码可靠性。 设计模 式使代码编写真正工程化;设计模式是软件工程的基石脉络,如同大厦的结构一样。

单例模式:

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

单例模式有两种实现模式:

饿汉模式:
  • 饿汉模式:就是说不管你将来用不用,程序启动时就创建一个唯一的实例对象,提前(main函数启动时)创建好实例对象
    • 优点:实现简单
    • 缺点:1、可能会导致进程启动慢、2、如果两个单例有启动先后顺序,那么饿汉无法控制
class A
{
public:static A * GetInstance(){return &_inst;}void Add(const string & key, const string & value){_dict[key] = value;}void Print(){for (auto& kv : _dict){cout << kv.first << ":" << kv.second << endl;}cout << endl;}
private:// 构造函数私有A(){}A(const A & aa) = delete;A& operator=(const A& aa) = delete;map<string, string> _dict;int _n = 0;static A _inst;
};// 在程序入口之前就完成单例对象的初始化
A A::_inst;int main()
{// 无法创建//A aa1;//A aa2;// 可以进行调用A::GetInstance()->Add("sort", "排序");A::GetInstance()->Add("left", "左边");A::GetInstance()->Add("right", "右边");A::GetInstance()->Print();// 拷贝构造禁止使用// A copy(*A::GetInstance());// copy.Print();// A::GetInstance()->Add("left", "剩余");// copy.Print();// A::GetInstance()->Print();// 赋值也无法使用// *A::GetInstance() = *A::GetInstance();return 0;
}
  • 如果这个单例对象在多线程高并发环境下频繁使用,性能要求较高,那么显然使用饿汉模式来避 免资源竞争,提高响应速度更好。
懒汉模式:
  • 如果单例对象构造十分耗时或者占用很多资源,比如加载插件啊, 初始化网络连接啊,读取 文件啊等等,而有可能该对象程序运行时不会用到,那么也要在程序一开始就进行初始化, 就会导致程序启动时非常的缓慢。 所以这种情况使用懒汉模式(延迟加载)更好。

    • 优点:第一次使用实例对象时,创建对象。进程启动无负载。多个单例实例启动顺序自由控制。
    • 缺点:复杂
class B
{
public:static B* GetInstance(){if (_inst == nullptr){_inst = new B;}return _inst;}void Add(const string& key, const string& value){_dict[key] = value;}void Print(){for (auto& kv : _dict){cout << kv.first << ":" << kv.second << endl;}cout << endl;}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;map<string, string> _dict;int _n = 0;static B* _inst;// 期望main函数结束后自动调用class gc{public:~gc(){DelInstance();}};static gc _gc;
};B* B::_inst = nullptr;// 期望main函数结束后自动调用
B::gc B::_gc;int main()
{B::GetInstance()->Add("sort", "排序");B::GetInstance()->Add("left", "左边");B::GetInstance()->Add("right", "右边");B::GetInstance()->Print();B::GetInstance()->Add("right", "xxx");B::GetInstance()->Print();// 显示的调用释放B::DelInstance();cout << "xxxxxxxxxxx" << endl;return 0;
}

懒汉模式的线程安全问题

  • 懒汉模式的线程安全问题已经在 Linux多线程【线程互斥】 中讲解:
    • 大概实现如下:

在这里插入图片描述

  • 局部的静态对象,是在第一次调用时初始化
  • C++11之前,他不是,也就说, C++11之前的编译器,那么这个代码不安全的
  • C++11之后可以保证局部静态对象的初始化是线程安全的,只初始化一次
class Singleton
{
public:// 提供获取单例对象的接口函数static Singleton& GetInstance(){// 局部的静态对象,是在第一次调用时初始化// C++11之前,他不是,也就说, C++11之前的编译器,那么这个代码不安全的// C++11之后可以保证局部静态对象的初始化是线程安全的,只初始化一次static Singleton inst;return inst;}private:// 构造函数私有Singleton(){cout << "Singleton()" << endl;}// 防拷贝Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton& s) = delete;
};int main()
{Singleton::GetInstance();return 0;
}
工厂模式
  • 用于创建对象的一个接口,让子类决定实例化哪个类
class Product {
public:virtual void show() = 0;
};class ConcreteProduct : public Product {
public:void show() override {std::cout << "ConcreteProduct Show" << std::endl;}
};class Factory {
public:virtual Product* createProduct() = 0;
};class ConcreteFactory : public Factory {
public:Product* createProduct() override {return new ConcreteProduct();}
};
观察者模式
  • 定义了一种一对多的依赖关系,让多个观察者对象同时监听某一主题对象
#include <iostream>
#include <list>
#include <string>class Observer {
public:virtual void update(std::string message) = 0;
};class ConcreteObserver : public Observer {
public:void update(std::string message) override {std::cout << "Received message: " << message << std::endl;}
};class Subject {
private:std::list<Observer*> observers;std::string message;public:void attach(Observer* observer) {observers.push_back(observer);}void notify() {for (auto observer : observers) {observer->update(message);}}void setMessage(std::string msg) {message = msg;}
};int main() {Subject subject;Observer* observer = new ConcreteObserver();subject.attach(observer);subject.setMessage("Hello, World!");subject.notify();delete observer;return 0;
}

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

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

相关文章

代发考生战报:6月25号 南宁 HCIP-Transmission传输 H31-341考试884分通过

代发考生战报&#xff1a;6月25号 南宁 HCIP-Transmission传输 H31-341考试884分通过 &#xff0c;今天我和同事两个人去考的&#xff0c;我考试遇到1个新题&#xff0c;他遇到两个新题&#xff0c;客服提供的题库很稳定&#xff0c;全覆盖了&#xff0c;轻松通过&#xff0c;考…

【语言模型】Xinference的部署过程

一、引言 Xinference&#xff0c;也称为Xorbits Inference&#xff0c;是一个性能强大且功能全面的分布式推理框架&#xff0c;专为各种模型的推理而设计。无论是研究者、开发者还是数据科学家&#xff0c;都可以通过Xinference轻松部署自己的模型或内置的前沿开源模型。Xinfe…

pikachu靶场 利用Rce上传一句话木马案例(工具:中国蚁剑)

目录 一、准备靶场&#xff0c;进入RCE 二、测试写入文件 三、使用中国蚁剑 一、准备靶场&#xff0c;进入RCE 我这里用的是pikachu 打开pikachu靶场&#xff0c;选择 RCE > exec "ping" 测试是否存在 Rce 漏洞 因为我们猜测在这个 ping 功能是直接调用系统…

性能评测系列:云架构扩展演进横向对比

原始测评报告 性能评测系列&#xff08;PT-010&#xff09;&#xff1a;Spring Boot RDS for MySQL&#xff0c;高并发insert 性能评测系列&#xff08;PT-012&#xff09;&#xff1a;Spring Boot(K8s多实例) RDS for MySQL&#xff0c;高并发insert 性能评测系列&#xff…

一元线性回归-R语言

# # 安装包 # install.packages(ggplot2) # library(ggplot2) Sys.setlocale(category LC_ALL, locale English_United States.1252) # Sys.setlocale("LC_ALL","Chinese") x <- c(18, 20, 22, 24, 26, 28, 30) y <- c(26.86, 28.35, 28.87,28.75,…

Linux——vim的配置文件+异常处理

vim的配置文件&#xff1a; [rootserver ~]# vim /etc/vimrc # 输入以下内容 set nu # 永久设置行号 shell [rootserver ~]# vim /etc/vimrc 或者 vim ~/.vimrc set hlsearch "高亮度反白 set backspace2 "可随时用退格键删除 set autoindent…

期货的杠杆怎么计算?

什么是杠杆系数 杠杆系数是指期货合约价值与保证金之间的比例。它表示投资者只需投入少量资金&#xff0c;就可以控制价值更高的期货合约。杠杆系数越高&#xff0c;投资者的资金放大倍数就越大&#xff0c;但风险也越大。 什么是期货保证金呢&#xff1f; 期货保证金&…

《HelloGitHub》第 99 期

兴趣是最好的老师&#xff0c;HelloGitHub 让你对编程感兴趣&#xff01; 简介 HelloGitHub 分享 GitHub 上有趣、入门级的开源项目。 github.com/521xueweihan/HelloGitHub 这里有实战项目、入门教程、黑科技、开源书籍、大厂开源项目等&#xff0c;涵盖多种编程语言 Python、…

Multicolor Dragon-MCD 六彩神龙_RSI

MCDX 六彩神龙 https://www.tradingview.com/script/u2dIgVpN-M2J-Indicator-MCDX/ MCDX is an indicator based on mutilple Relative Strength Index (RSI) with different period, then classify into 3 categories - Retailer, Hot Money and Banker - Green - Retailer零…

2024.06.28 刷题日记

394. 字符串解码 给定一个经过编码的字符串&#xff0c;返回它解码后的字符串。 示例 1&#xff1a; 输入&#xff1a;s “3[a]2[bc]” 输出&#xff1a;“aaabcbc” 示例 2&#xff1a; 输入&#xff1a;s “3[a2[c]]” 输出&#xff1a;“accaccacc” 示例 3&#xff1a;…

怎么进行模型微调,以微调llama3为例

微调模型&#xff08;Fine-tuning&#xff09;通常涉及以下步骤&#xff0c;以微调 LLaMA 3 为例&#xff1a; 1. 准备工作 在开始微调之前&#xff0c;需要准备以下工作&#xff1a; 选择预训练模型&#xff1a;LLaMA 3 是一个大型的语言模型&#xff0c;可以通过 Hugging F…

react 中 Swiper vertical 模式下 autoHeight 失效的问题

Swiper 在 vertical 模式下 autoHeight 失效的问题&#xff0c;导致页面出现多余的空白高度&#xff0c;网上找了很多方法都无效&#xff0c;我直接暴力更改。 <SwiperclassNameindex-swiperdirection{vertical}mousewheel{true}centeredSlides{true}autoHeight{true}slide…

VS2019+QT5.12.10: error MSB4036: 未找到“Join”任务。请检查下列各项: 1.) 项目文件中的任务名

1、背景 两个VS2019打开两个相同的项目&#xff0c;一个里可以正常运行&#xff0c; 一个中一直报错&#xff0c;&#xff0c;报的错也是瞎几把报的。。 2、重新安装插件 之前在VS的扩展中在线安装了qt插件&#xff0c; 安装了一半&#xff0c;比较慢&#xff0c;直接强行退出…

传媒行业指哪些?需要过等保吗?

传媒&#xff0c;一个人人都接触的行业。相信大家都听过传媒&#xff0c;但具体传媒行业是指什么&#xff0c;包括哪些&#xff0c;详细很多人都不了解。这不一些人在问&#xff0c;传媒行业指哪些&#xff1f;需要过等保吗&#xff1f;这里跟我们小编一起来讨论讨论吧&#xf…

玩游戏就能学习亚马逊云科技AWS技术并通过热门技术认证考试??

亚马逊AWS限时活动&#xff0c;玩免费游戏Cloud Quest Practitioner送AWS云从业证书考试25%折扣券(价值171元)&#xff0c;玩游戏的同时还能学知识一举两得。Cloud Quest是AWS出的一款3D角色扮演游戏/虚拟城市建造形式的实验课程(游戏画面有点像天际线)&#xff0c;大家通过完成…

【01-02】Mybatis的配置文件与基于XML的使用

1、引入日志 在这里我们引入SLF4J的日志门面&#xff0c;使用logback的具体日志实现&#xff1b;引入相关依赖&#xff1a; <!--日志的依赖--><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version&g…

338. 比特位计数(leetcode)

338. 比特位计数&#xff08;leetcode&#xff09; 题目描述 给你一个整数 n &#xff0c;对于 0 < i < n 中的每个 i &#xff0c;计算其二进制表示中 1 的个数 &#xff0c;返回一个长度为 n 1 的数组 ans 作为答案。 示例1 输入&#xff1a;n 2 输出&#xff1a;[0…

Sorting

本节提供有关在数据网格中对数据进行排序的信息。 GridControl-Grid View Sort Data 默认情况下&#xff0c;最终用户可以按任何列对数据进行排序&#xff0c;但使用MemoExEdit、ImageEdit和PictureEdit在位编辑器的列除外。在运行时&#xff0c;单击列标题一次以升序排列数…

中国电信股份有限公司江西分公司招聘信息 7.5日截止

法律事务管理(南昌) 学历要求 本科及以上学历 岗位职责 1.依据国家法律、法规和相关规章规定,为公司其他部门提供日常法律服务与支持; 2.负责公司各类合同审核工作; 3.负责公司法律文件的起草和法律事务谈判; 4.围绕与公司业务有关的法律问题及法…

如何在Java中实现分布式缓存?

如何在Java中实现分布式缓存&#xff1f; 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天我们将深入探讨在Java应用程序中如何实现分布式缓存&#xff0c;探…