当前位置: 首页 > news >正文

利用C++11和泛型编程改进原型模式

引言

在软件开发中,原型模式是一种常用的设计模式,主要用于创建对象的克隆。通过原型模式,我们可以避免复杂的对象创建过程,尤其是当对象的初始化需要大量资源或复杂操作时。本文将通过一个具体的例子,详细介绍如何在C++中使用C++11和泛型编程来实现原型模式,并探讨其在面向对象设计原则中的应用。

实现步骤

1. 定义基类

首先,我们需要定义一个基类Prototype,它将作为所有具体行为类的父类。基类中包含两个纯虚函数:clone()execute()

#include <memory>
#include <string>
#include <typeindex>
#include <unordered_map>template <typename T>
class Prototype {
public:virtual ~Prototype() = default;virtual std::shared_ptr<Prototype<T>> clone() const = 0;virtual void execute() const = 0;
};
  • clone():用于克隆当前对象,返回一个指向新对象的智能指针。
  • execute():用于展示具体的行为,每个子类将根据自己的行为实现该方法。

2. 创建具体行为类

接下来,我们创建三个具体的行为类,分别继承自Prototype,并实现各自的clone()execute()方法。

拿着饭碗吃饭的行为类

class EatYongYong : public Prototype<EatYongYong> {
public:std::shared_ptr<Prototype<EatYongYong>> clone() const override {return std::make_shared<EatYongYong>(*this);}void execute() const override {std::cout << "勇勇拿着饭碗在吃饭。" << std::endl;}
};

打压下属的行为类

class PressYongYong : public Prototype<PressYongYong> {
public:std::shared_ptr<Prototype<PressYongYong>> clone() const override {return std::make_shared<PressYongYong>(*this);}void execute() const override {std::cout << "勇勇在打压下属。" << std::endl;}
};

欺骗领导的行为类

class DeceiveYongYong : public Prototype<DeceiveYongYong> {
public:std::shared_ptr<Prototype<DeceiveYongYong>> clone() const override {return std::make_shared<DeceiveYongYong>(*this);}void execute() const override {std::cout << "勇勇在欺骗领导。" << std::endl;}
};

3. 创建管理类

为了管理不同行为的YongYong对象,我们创建一个PrototypeManager类。该类使用std::unordered_map存储不同行为的原型对象,并提供获取原型对象的方法。

template <typename... Types>
class PrototypeManager {
private:using PrototypeMap = std::unordered_map<std::type_index, std::shared_ptr<std::decay_t<Types>>...>;PrototypeMap prototypes;public:template <typename T, typename... Args>void registerPrototype(const Args&... args) {prototypes[std::type_index typeid(T)] = std::make_shared<T>(args...);}template <typename T>std::shared_ptr<T> getPrototype() const {auto it = prototypes.find(std::type_index typeid(T));if (it != prototypes.end()) {return std::dynamic_pointer_cast<T>(it->second->clone());}return nullptr;}
};

4. 使用管理类创建对象

main函数中,我们使用PrototypeManager创建不同行为的YongYong对象,并调用它们的execute()方法。

int main() {PrototypeManager<EatYongYong, PressYongYong, DeceiveYongYong> manager;manager.registerPrototype<EatYongYong>();manager.registerPrototype<PressYongYong>();manager.registerPrototype<DeceiveYongYong>();auto yongYong1 = manager.getPrototype<EatYongYong>();yongYong1->execute();auto yongYong2 = manager.getPrototype<PressYongYong>();yongYong2->execute();auto yongYong3 = manager.getPrototype<DeceiveYongYong>();yongYong3->execute();return 0;
}

代码解释

基类Prototype

  • ~Prototype() = default;:提供默认的析构函数,确保子类对象能够正确析构。
  • clone():纯虚函数,定义克隆方法,子类必须实现。
  • execute():纯虚函数,定义行为方法,子类必须实现。

具体行为类

每个具体行为类(如EatYongYong)都继承自Prototype,并实现以下方法:

  • clone()方法:返回自身对象的克隆。
  • execute()方法:展示具体行为。

管理类PrototypeManager

  • prototypes:使用std::unordered_map存储不同行为的原型对象,键为std::type_index,值为对应行为的智能指针。
  • registerPrototype()方法:允许在运行时动态注册新的行为类,并支持传入构造参数。
  • getPrototype()方法:根据类型返回对应的原型对象的克隆。

面向对象设计原则分析

  1. 单一职责原则(SRP)
  • 基类Prototype:职责明确,定义克隆和行为方法。
  • 具体行为类:每个类只实现一个具体行为。
  • 管理类PrototypeManager:职责单一,管理原型对象。
  1. 开闭原则(OCP)
  • 基类Prototype:允许子类扩展行为,无需修改基类。
  • 管理类PrototypeManager:通过添加新的行为类扩展功能,无需修改现有代码。
  1. 里氏替换原则(LSP)
  • 具体行为类:可以替换基类Prototype对象,确保代码正确性。
  1. 依赖倒置原则(DIP)
  • 管理类PrototypeManager:依赖于抽象的Prototype接口,而不是具体实现。
  1. 接口隔离原则(ISP)
  • 基类Prototype:定义清晰的接口,无冗余方法。
  • 具体行为类:只实现需要的方法。
  1. 合成复用原则(CRP)
  • 管理类PrototypeManager:通过组合管理对象,而不是通过继承复用代码。

总结

通过使用C++11和泛型编程技术,改进后的原型模式实现具有以下优势:

  1. 类型安全:使用std::type_index和模板参数,避免了原始代码中的字符串键和类型转换问题。
  2. 灵活性:支持任意数量的行为类,扩展性更强。
  3. 性能:编译时类型检查和std::unordered_map的使用,提高了运行效率。
  4. 可维护性:代码更加模块化,新增行为类更加简单。

这种改进后的原型模式实现,能够更好地满足现代C++开发需求,同时保持代码的简洁和高效。希望本文能够帮助读者更好地理解和应用原型模式以及相关的面向对象设计原则。

http://www.xdnf.cn/news/1252495.html

相关文章:

  • .Net下载共享文件夹中的文件
  • Java Stream API 详解(Java 8+)
  • Linux---第二天---基础指令
  • 快速莫比乌斯变换(FMT)与莫比乌斯反演 例题:树上lcm
  • SELinux 安全机制详解与管理
  • 组合期权:跨式策略
  • 批量提问程序开发方案:基于Python的百度文小言接口实现
  • 基于 Jenkins Pipeline 实现 DITA 文档自动化构建与发布(开源方案)
  • 百度智能云给“数字人”发工牌
  • Boosting 知识点整理:调参技巧、可解释性工具与实战案例
  • Bug 记录:SecureRandom.getInstanceStrong()导致验证码获取阻塞
  • 【motion】标签体系设计与检索 1:HumanML3D 和 KIT Motion-Language(KITML)
  • Java 使用动态代理和反射实现字段变更跟踪
  • 生成网站sitemap.xml地图教程
  • 【STM32U385RG 测评】基于VSCode的STM32开发环境搭建
  • 西门子PLC基础指令6:读取时钟指令、设置时钟指令、使能含义与注意
  • 【32】C++实战篇—— m行n列的坐标点,求每行相邻点X差值dX,每列相邻点y差值dY,并以矩阵形式左端对齐
  • JAVA--流程控制语句
  • 【VS + Qt】VS2022 Qt 开发中 ui_xx.h 文件编辑报错但编译正常的问题解决
  • 「iOS」————单例与代理
  • 如何解决pip安装报错ModuleNotFoundError: No module named ‘caffe’问题
  • 河南萌新联赛2025第四场-河南大学
  • K8S云原生监控方案Prometheus+grafana
  • yolov1-v3原理解析
  • DHCP 服务器与DNS服务器
  • 服务器——“查询不到显卡驱动,且输入nvidia-smi报错”的解决办法
  • 2.6 sync
  • 媒体资产管理系统和OCR文字识别的结合
  • 多端同步新解法:Joplin+cpolar联合通过开源设计实现跨平台无缝协作?
  • 自动驾驶系统的网络安全风险分析