接口类和抽象类在设计模式中的一些应用

C++设计模式中,有些模式需要使用接口类(Interface Class)和抽象类(Abstract Class)来实现特定的设计目标。以下是一些常见的设计模式及其需要的原因,并附上相应的代码片段。

1. 策略模式(Strategy Pattern

需要接口类:策略模式通过定义一组算法或行为接口,使得算法可以独立于使用它们的客户端而变化。

原因:策略模式需要定义一组通用的行为接口,这些接口可以在运行时动态切换。

代码片段:

// 接口类
class Strategy {
public:virtual void execute() = 0;  // 纯虚函数
};// 具体策略类
class ConcreteStrategyA : public Strategy {
public:void execute() override {std::cout << "Executing strategy A\n";}
};class ConcreteStrategyB : public Strategy {
public:void execute() override {std::cout << "Executing strategy B\n";}
};// 上下文类
class Context {
private:Strategy* strategy;public:Context(Strategy* s) : strategy(s) {}void setStrategy(Strategy* s) {strategy = s;}void executeStrategy() {strategy->execute();}
};// 使用策略模式
int main() {Context context(new ConcreteStrategyA());context.executeStrategy();  // 输出 "Executing strategy A"context.setStrategy(new ConcreteStrategyB());context.executeStrategy();  // 输出 "Executing strategy B"return 0;
}

2. 工厂方法模式(Factory Method Pattern

需要抽象类:工厂方法模式通过定义一个创建对象的接口,但将具体类的实例化延迟到子类中。

原因:工厂方法模式需要定义一个抽象类,用于定义创建对象的接口,并让子类实现具体的产品创建逻辑。

代码片段:

// 抽象产品类
class Product {
public:virtual void use() = 0;  // 纯虚函数
};// 具体产品类
class ConcreteProductA : public Product {
public:void use() override {std::cout << "Using product A\n";}
};class ConcreteProductB : public Product {
public:void use() override {std::cout << "Using product B\n";}
};// 抽象工厂类
class Creator {
public:virtual Product* factoryMethod() = 0;  // 纯虚函数
};// 具体工厂类
class ConcreteCreatorA : public Creator {
public:Product* factoryMethod() override {return new ConcreteProductA();}
};class ConcreteCreatorB : public Creator {
public:Product* factoryMethod() override {return new ConcreteProductB();}
};// 使用工厂方法模式
int main() {Creator* creator = new ConcreteCreatorA();Product* product = creator->factoryMethod();product->use();  // 输出 "Using product A"creator = new ConcreteCreatorB();product = creator->factoryMethod();product->use();  // 输出 "Using product B"return 0;
}

3. 观察者模式(Observer Pattern

需要接口类:观察者模式通过定义观察者和主题的接口,使得主题状态发生变化时,所有观察者都能得到通知。

原因:观察者模式需要定义通用的观察者和主题接口,以便在运行时动态添加和移除观察者。

代码片段:

// 接口类:观察者
class Observer {
public:virtual void update(int value) = 0;  // 纯虚函数
};// 具体观察者类
class ConcreteObserverA : public Observer {
public:void update(int value) override {std::cout << "Observer A received update: " << value << "\n";}
};class ConcreteObserverB : public Observer {
public:void update(int value) override {std::cout << "Observer B received update: " << value << "\n";}
};// 主题类
class Subject {
private:std::vector<Observer*> observers;int state;public:void attach(Observer* observer) {observers.push_back(observer);}void setState(int value) {state = value;notifyObservers();}void notifyObservers() {for (Observer* observer : observers) {observer->update(state);}}
};// 使用观察者模式
int main() {Subject subject;ConcreteObserverA observerA;ConcreteObserverB observerB;subject.attach(&observerA);subject.attach(&observerB);subject.setState(10);  // 输出 "Observer A received update: 10" 和 "Observer B received update: 10"return 0;
}

4. 模板方法模式(Template Method Pattern

需要抽象类:模板方法模式通过定义一个算法的骨架,将某些步骤延迟到子类中实现。

原因:模板方法模式需要定义一个抽象类,用于定义算法的骨架,并让子类实现具体的步骤。

代码片段:

// 抽象类
class AbstractClass {
public:void templateMethod() {step1();step2();step3();}virtual void step1() = 0;  // 纯虚函数virtual void step2() = 0;  // 纯虚函数void step3() {std::cout << "Step 3 in AbstractClass\n";}
};// 具体子类
class ConcreteClassA : public AbstractClass {
public:void step1() override {std::cout << "Step 1 in ConcreteClassA\n";}void step2() override {std::cout << "Step 2 in ConcreteClassA\n";}
};class ConcreteClassB : public AbstractClass {
public:void step1() override {std::cout << "Step 1 in ConcreteClassB\n";}void step2() override {std::cout << "Step 2 in ConcreteClassB\n";}
};// 使用模板方法模式
int main() {AbstractClass* obj1 = new ConcreteClassA();AbstractClass* obj2 = new ConcreteClassB();obj1->templateMethod();// 输出 "Step 1 in ConcreteClassA"// 输出 "Step 2 in ConcreteClassA"// 输出 "Step 3 in AbstractClass"obj2->templateMethod();// 输出 "Step 1 in ConcreteClassB"// 输出 "Step 2 in ConcreteClassB"// 输出 "Step 3 in AbstractClass"delete obj1;delete obj2;return 0;
}

在C++设计模式中,命令模式(Command Pattern)、状态模式(State Pattern)、职责链模式(Chain of Responsibility Pattern)和组合模式(Composite Pattern)也都涉及使用接口类和抽象类来实现特定的设计目标。以下是对这些模式的详细说明,并附上相应的代码片段。

1. 命令模式(Command Pattern

需要接口类:命令模式通过定义命令的接口,将请求封装为对象,使得可以参数化客户端对象、记录请求队列、支持撤销操作等。

原因:命令模式需要定义一个通用的命令接口,使得不同的命令对象可以被统一处理。

代码片段:
 

// 接口类:命令
class Command {
public:virtual void execute() = 0;  // 纯虚函数
};// 具体命令类
class ConcreteCommandA : public Command {
private:std::string recipient;public:ConcreteCommandA(const std::string& r) : recipient(r) {}void execute() override {std::cout << "Command A executed by " << recipient << "\n";}
};class ConcreteCommandB : public Command {
private:std::string recipient;public:ConcreteCommandB(const std::string& r) : recipient(r) {}void execute() override {std::cout << "Command B executed by " << recipient << "\n";}
};// 调用者类
class Invoker {
private:Command* command;public:void setCommand(Command* c) {command = c;}void executeCommand() {command->execute();}
};// 使用命令模式
int main() {Invoker invoker;ConcreteCommandA commandA("Client A");ConcreteCommandB commandB("Client B");invoker.setCommand(&commandA);invoker.executeCommand();  // 输出 "Command A executed by Client A"invoker.setCommand(&commandB);invoker.executeCommand();  // 输出 "Command B executed by Client B"return 0;
}

2. 状态模式(State Pattern

需要接口类:状态模式通过定义状态的接口,使得对象的行为可以根据其内部状态的改变而改变。

原因:状态模式需要定义一个通用的状态接口,使得不同的状态对象可以被统一处理。

代码片段:

// 接口类:状态
class State {
public:virtual void handle() = 0;  // 纯虚函数
};// 具体状态类
class ConcreteStateA : public State {
public:void handle() override {std::cout << "Handling state A\n";}
};class ConcreteStateB : public State {
public:void handle() override {std::cout << "Handling state B\n";}
};// 上下文类
class Context {
private:State* state;public:void setState(State* s) {state = s;}void request() {state->handle();}
};// 使用状态模式
int main() {Context context;ConcreteStateA stateA;ConcreteStateB stateB;context.setState(&stateA);context.request();  // 输出 "Handling state A"context.setState(&stateB);context.request();  // 输出 "Handling state B"return 0;
}

3. 职责链模式(Chain of Responsibility Pattern

需要接口类:职责链模式通过定义处理请求的接口,将多个处理者链接在一起,使得请求可以沿着链进行传递,直到被处理为止。

原因:职责链模式需要定义一个通用的处理者接口,使得不同的处理者可以被统一处理。

代码片段:

// 接口类:处理者
class Handler {
protected:Handler* next;public:void setNext(Handler* n) {next = n;}virtual void handleRequest(int request) = 0;  // 纯虚函数
};// 具体处理者类
class ConcreteHandlerA : public Handler {
public:void handleRequest(int request) override {if (request < 10) {std::cout << "ConcreteHandlerA handled request " << request << "\n";} else if (next != nullptr) {next->handleRequest(request);}}
};class ConcreteHandlerB : public Handler {
public:void handleRequest(int request) override {if (request >= 10 && request < 20) {std::cout << "ConcreteHandlerB handled request " << request << "\n";} else if (next != nullptr) {next->handleRequest(request);}}
};// 使用职责链模式
int main() {ConcreteHandlerA handlerA;ConcreteHandlerB handlerB;handlerA.setNext(&handlerB);handlerA.handleRequest(5);   // 输出 "ConcreteHandlerA handled request 5"handlerA.handleRequest(15);  // 输出 "ConcreteHandlerB handled request 15"handlerA.handleRequest(25);  // 没有处理者处理该请求return 0;
}

4. 组合模式(Composite Pattern

需要抽象类:组合模式通过定义组合对象和叶子对象的接口,使得客户端可以统一处理单个对象和组合对象。

原因:组合模式需要定义一个通用的组件接口,使得叶子节点和组合节点可以被统一处理。

代码片段:

// 抽象类:组件
class Component {
public:virtual void operation() = 0;  // 纯虚函数virtual void add(Component* component) {}virtual void remove(Component* component) {}virtual Component* getChild(int index) { return nullptr; }
};// 叶子类
class Leaf : public Component {
public:void operation() override {std::cout << "Leaf operation\n";}
};// 组合类
class Composite : public Component {
private:std::vector<Component*> children;public:void operation() override {std::cout << "Composite operation\n";for (Component* child : children) {child->operation();}}void add(Component* component) override {children.push_back(component);}void remove(Component* component) override {children.erase(std::remove(children.begin(), children.end(), component), children.end());}Component* getChild(int index) override {return children[index];}
};// 使用组合模式
int main() {Composite root;Leaf leaf1;Leaf leaf2;Composite branch;branch.add(&leaf1);branch.add(&leaf2);root.add(&branch);root.operation();// 输出 "Composite operation"// 输出 "Leaf operation"// 输出 "Leaf operation"return 0;
}

总结

  • 策略模式:需要接口类来定义一组通用的行为接口。
  • 工厂方法模式:需要抽象类来定义创建对象的接口,并让子类实现具体的产品创建逻辑。
  • 观察者模式:需要接口类来定义观察者和主题的接口,以便在运行时动态添加和移除观察者。
  • 模板方法模式:需要抽象类来定义算法的骨架,并让子类实现具体的步骤。
  • 命令模式:需要接口类来定义命令的接口,使得可以参数化客户端对象、记录请求队列、支持撤销操作等。
  • 状态模式:需要接口类来定义状态的接口,使得对象的行为可以根据其内部状态的改变而改变。
  • 职责链模式:需要接口类来定义处理请求的接口,使得请求可以沿着链进行传递,直到被处理为止。
  • 组合模式:需要抽象类来定义组合对象和叶子对象的接口,使得客户端可以统一处理单个对象和组合对象。

这些设计模式通过接口类和抽象类,提供了灵活的、可扩展的解决方案,有效地解决了软件设计中的常见问题。

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

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

相关文章

linux rocky 9.4部署和管理docker harbor私有源

文章目录 Harbor简介安装Harbor技术细节1.安装系统(略),设置主机名和IP2.安装docker3.安装docker-compose4.安装Harbor私有源仓库5 测试登录1.本机登录2.客户端登录Harbor服务器配置docker源1. 下载镜像2.把镜像上传到Harbor私有仓库源3.客户端下载镜像,并且启动容器linux …

【Elasticsearch入门到落地】1、初识Elasticsearch

一、什么是Elasticsearch Elasticsearch&#xff08;简称ES&#xff09;是一款非常强大的开源搜索引擎&#xff0c;可以帮助我们从海量数据中快速找到需要的内容。它使用Java编写&#xff0c;基于Apache Lucene来构建索引和提供搜索功能&#xff0c;是一个分布式、可扩展、近实…

【算法一周目】双指针(2)

目录 有效三角形的个数 解题思路 C代码实现 和为s的两个数字 解题思路 C代码实现 三数之和 解题思路 C代码实现 四数之和 解题思路 C代码实现 有效三角形的个数 题目链接&#xff1a;611. 有效三角形的个数题目描述&#xff1a;给定一个包含非负整数的数组nums&…

单体架构 IM 系统之 Server 节点状态化分析

基于 http 短轮询模式的单体架构的 IM 系统见下图&#xff0c;即客户端通过 http 周期性地轮询访问 server 实现消息的即时通讯&#xff0c;也就是我们前面提到的 “信箱模型”。“信箱模型” 虽然实现非常容易&#xff0c;但是消息的实时性不高。 我们在上一篇文章&#xff08…

让AI为你发声!Windows电脑快速部署ChatTTS文本转语音神器

文章目录 前言1. 下载运行ChatTTS模型2. 安装Cpolar工具3. 实现公网访问4. 配置ChatTTS固定公网地址 前言 嘿&#xff0c;朋友们&#xff01;今天我们来聊聊如何在Windows系统上快速搭建ChatTTS&#xff0c;一个超酷的开源文本转语音项目。更棒的是&#xff0c;我们还可以用Cp…

RSTP的配置

RSTP相对于STP在端口角色、端口状态、配置BPDU格式、配置BPDU的处理方式、快速收敛机制、拓扑变更机制和4种保护特性方面的详细改进说明&#xff1a; 端口角色&#xff1a; STP中定义了三种端口角色&#xff1a;根端口&#xff08;Root Port&#xff09;、指定端口&#xff0…

elementui el-table中给表头 el-table-column 加一个鼠标移入提示说明

前言 在使用el-table 表格中有些表格的表头需要加入一些提示&#xff0c;鼠标移入则出现提示&#xff0c;非常实用&#xff0c;我是通过el-table中的el-tooltip实现的&#xff0c;以下的效果预览 代码实现 <el-table ref"multipleTable" :data"data"…

ubuntu18.04 安装与卸载NCCL conda环境安装PaddlePaddle

cuda版本11.2 说明PaddlePaddle需要安装NCCL 1、Log in | NVIDIA Developer 登录官网 找到对应版本 官方提供了多种安装方式&#xff0c;本文使用Local installers (x86)本地安装 点击对应的版本下载如&#xff1a; nccl-local-repo-ubuntu1804-2.8.4-cuda11.2_1.0-1_amd6…

机器学习—决定下一步做什么

现在已经看到了很多不同的学习算法&#xff0c;包括线性回归、逻辑回归甚至深度学习或神经网络。 关于如何构建机器学习系统的一些建议 假设你已经实现了正则化线性回归来预测房价&#xff0c;所以你有通常的学习算法的成本函数平方误差加上这个正则化项&#xff0c;但是如果…

【Rust中的项目管理】

Rust中的项目管理 前言Package&#xff0c;Crate&#xff0c;Module &use &#xff0c;Path通过代码示例解释 Crate&#xff0c;Module &#xff0c;use&#xff0c;Path创建一个package&#xff1a;代码组织化skin.rs 中的代码struct & enum 相对路径和绝对路径引用同…

labview用sql server数据库存取数据到一个单元格

最近有一个项目上需要一个庞大的数据量&#xff0c;需要很多列&#xff0c;但是百度查了一下sqi server最多支持1024列&#xff0c;这一限制适用于大多数表类型&#xff0c;包括常规表&#xff0c;临时表和表变量&#xff0c;要注意的是如果超出这一限制可能会导致数据的完整性…

架构篇(04理解架构的演进)

目录 学习前言 一、架构演进 1. 初始阶段的网站架构 2. 应用服务和数据服务分离 3. 使用缓存改善网站性能 4. 使用应用服务器集群改善网站的并发处理能力 5. 数据库读写分离 6. 使用反向代理和CDN加上网站相应 7. 使用分布式文件系统和分布式数据库系统 8. 使用NoSQL和…

Unity学习笔记(4):人物和基本组件

文章目录 前言开发环境新增角色添加组件RigidBody 2D全局项目设置Edit 给地图添加碰撞体 总结 前言 今天不加班&#xff0c;有空闲时间。争取一天学一课&#xff0c;养成习惯 开发环境 Unity 6windows 11vs studio 2022Unity2022.2 最新教程《勇士传说》入门到进阶&#xff…

Java项目实战II基于Spring Boot的高校教师电子名片系统的设计与实现(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。 一、前言 在信息化教育日益普及的今天&#xff0…

数学几百年重大错误:将无穷多各异直线误为直线y=x

黄小宁 h定理&#xff1a;点集AB≌B的必要条件是A≌B。 证&#xff1a;若AB则A必可恒等变换地变为BA≌A&#xff0c;而恒等变换是保距变换。证毕。 直线Z&#xff1a;x-y0&#xff08;x的变域是x轴&#xff09;可放大&#xff08;拉伸&#xff09;变换为直线L&#xff08;不≌Z…

学习threejs,使用第一视角控制器FirstPersonControls控制相机

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言1.1 ☘️第一视角控制器FirstPerson…

LabVIEW导入并显示CAD DXF文件图形 程序见附件

LabVIEW导入并显示CAD DXF文件图形 程序见附件 LabVIEW导入并显示CAD DXF文件图形 程序见附件 - 北京瀚文网星科技有限公司 LabVIEW广泛应用于自动化、数据采集、图形显示等领域。对于涉及CAD图形的应用&#xff0c;LabVIEW也提供了一些方法来导入和显示CAD DXF文件&#x…

数据结构---详解栈

一、栈的概念和结构 栈&#xff1a;⼀种特殊的线性表&#xff0c;其只允许在固定的⼀端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶&#xff0c;另一端称为栈底。栈中的数据元素遵守后进先出LIFO&#xff08;Last In First Out&#xff09;的原则。 压栈&a…

Python Plotly 库使用教程

Python Plotly 库使用教程 引言 数据可视化是数据分析中至关重要的一部分&#xff0c;它能够帮助我们更直观地理解数据、发现潜在的模式和趋势。Python 提供了多种数据可视化库&#xff0c;其中 Plotly 是一个功能强大且灵活的库&#xff0c;支持交互式图表的创建。与静态图表…

怎么样绑定域名到AWS(亚马逊云)服务器

1&#xff0c;拿着你买的域名去亚马逊申请一个证书。申请证书分两种&#xff0c;一种是去亚马逊后台填域名手动申请 &#xff0c;另一种是通过API来申请&#xff0c;类似如下代码&#xff1a; 2、证验证书。有两种方式&#xff1a;一种是通过邮件&#xff0c;另一种去到域名提供…