【深入浅出设计模式--命令模式】

深入浅出设计模式--命令模式

  • 一、背景
  • 二、问题
  • 三、解决方案
  • 四、试用场景总结
  • 五、后记

一、背景

  命令模式是一种行为设计模式,它可以将用户的命令请求转化为一个包含有相关参数信息的对象,命令的发送者不需要知道接收者是如何处理这条命令,多个功能入口可以发送同一命令,避免多处多次实现相同功能的冗余代码。另外可以对命令进行延迟处理,或放入队列或栈中,支持命令回撤。

二、问题

  在使用GUI界面的应用程序时,一般保存功能会有多个入口,比如点击按钮保存、点击菜单项保存和使用键盘ctrl+s保存等,每个入口的位置不同且没有什么关联关系,只是最终实现文档保存功能的代码是一样的。
在这里插入图片描述
  此时要么将这些操作代码复制粘贴进多个类中,要么就是让菜单项依赖于按钮,显然这两种方式都是不明智的。

三、解决方案

  一般在这种一对多、多对多的场景下,最好的方式是添加一个中间层,上层负责GUI的交互,下层负责业务逻辑的处理,中间层则将命令请求抽象为一个对象,在上下两层中传递消息数据,该对象可以连接不同的GUI和业务逻辑对象GUI 对象无需了解业务逻辑对象是否获得了请求,也无需了解其对请求进行处理的方式。
在这里插入图片描述
  1. 该模式的类图如下所示:发送者ICommandSender中包含了命令接口ICommand的指针,ICommand中只有execute虚函数,具体的命令类SaveCommand中包含有命令接收者Document的指针,且重写execute函数。CommandHistory类记录了每条执行了的命令,调用pop弹出命令时可以调用该命令的redo函数(下图中未展示)进行命令撤销。
在这里插入图片描述

  2. 该模式下各类之间交互的时序图如下:首先生成命令接收者Document,然后生成命令SaveCommand并绑定Document,接下来生成命令发送者Button并绑定SaveCommand,最后Button调用触发函数click让命令对象去执行具体的execute函数。
在这里插入图片描述
  3. 相关代码实现

  • 命令发送者
class CommandHistory{
public:static CommandHistory* GetInstance(){static CommandHistory cmdHis;return &cmdHis;}void push(ICommand* cmd){m_cmdHisVec.push(cmd);}void pop(){m_cmdHisVec.pop();}size_t size(){return m_cmdHisVec.size();}
private:stack<ICommand*> m_cmdHisVec;
};#define CMD_HIS CommandHistory::GetInstance()class ICommandSender{
public:ICommandSender(){}~ICommandSender(){}void setCommand(ICommand* cmd){m_cmd = cmd;}ICommand* getCommand(){return m_cmd;}
protected:ICommand* m_cmd;
};class Button : public ICommandSender{
public:void click(){m_cmd->execute("Sent by Button");CMD_HIS->push(m_cmd);cout << "Count of Button history command is " << CMD_HIS->size() << endl;}
};class Shortcut : public ICommandSender{
public:void setCommand(const string& key, ICommand* cmd){m_cmd = cmd;m_keyCmdMap[key] = cmd;}void press(){m_cmd->execute("Sent by Shortcut");CMD_HIS->push(m_cmd);cout << "Count of Shortcut history command is " << CMD_HIS->size() << endl;}
private:map<string, ICommand*> m_keyCmdMap;
};
  • 命令
class ICommand{
public:virtual ~ICommand(){}virtual void execute(const string& from) = 0;
};
class SaveCommand : public ICommand{
public:SaveCommand(Document* doc) : m_doc(doc) {}virtual void execute(const string& from) override{m_doc->setText(from);m_doc->save();}
private:Document* m_doc;
};
  • 命令接收者
class Document{
public:void setText(const string& text){m_text = text;}void save(){cout << m_text << " has been saved" << endl;               }
private:string m_text;
};

四、试用场景总结

  • 多个不同操作对应同一处理结果,那么可以使用命令模式。如点击菜单项、点击按钮和ctrl+s进行保存
  • 同一操作在不同的场景下产生不同的结果,在运行时切换已连接的命令,也可以使用命令模式。如用户可以配置菜单项,在点击时触发不同的命令。
  • 若需要将操作放入队列中,延迟或计划发送命令,也可以使用命令模式。
  • 若要实现操作回滚(撤销)功能,也可以使用命令模式。命令历史记录是一种包含所有已执行命令对象及其相关程序状态备份的栈结构。
  • 若要远程执行命令,需要将命令对象序列化,从而能方便地写入文件或数据库中。

五、后记

以上所有内容均为原创,代码已上传至gayhub:
https://github.com/gangster-puppy/Design-Pattern.git

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

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

相关文章

php文件上传功能(文件上传)

实现文件上传是Web开发中常用的功能之一&#xff0c;而PHP也是支持文件上传的。那么&#xff0c;下面我们就来介绍一下常用的PHP实现文件上传的方法。 使用HTML表单实现文件上传 HTML表单是Web开发中最基本的元素之一&#xff0c;它可以接收用户输入的数据&#xff0c;并通过…

大数据时代,数据治理

一、大数据时代还需要数据治理吗&#xff1f; 数据平台发展过程中随处可见的数据问题 大数据不是凭空而来&#xff0c;1981年第一个数据仓库诞生&#xff0c;到现在已经有了近40年的历史&#xff0c;相对数据仓库来说我还是个年轻人。而国内企业数据平台的建设大概从90年代末…

Docker 自动化部署(保姆级教程)

Docker 自动化部署 1. jenkins 介绍1.1 参考链接&#xff1a;1.2 jenkins 概述1.3 jenkins部署项目的流程 2. jenkins 安装2.1 基于docker 镜像2.2 启动 jenkins 后端服务2.3 登录 jenkins 服务后端 3. jenkins自动化部署开始3.1 下载需要的插件3.2 创建任务3.2.1 描述3.2.2 配…

使用bash脚本编译Qt工程

最近在搭建Qt工程的编译服务器&#xff0c;需要通过shell脚本执行工程的编译过程&#xff0c;写在这里&#xff0c;总结最近的工作。 1. 构建过程 以Windows为例&#xff0c;在QtCreator中左侧选择“项目”便可以看到编译的指令和参数&#xff0c;这些操作也会在我们点击“构建…

CAN总线数据链路层

CAN报文帧种类: 数据帧:用于发送节点向接收节点传送数据,是使用最多的帧类型远程帧:用于接收节点向某个发送节点请求数据错误帧:用于当某节点检测出错误时向其他节点通知错误的帧过载帧:用于接收节点向发送节点通知自身接收能力的帧帧间隔:将数据帧或远程帧与前面的帧分…

ElementUI之首页导航与左侧菜单

目录 一、Mock 1.1 什么是Mock.js 1.2 安装与配置 1.2.1 安装mock.js 1.2.2 引入mock.js 1.3 mock.js使用 1.3.1 定义测试数据文件 1.3.2 mock拦截Ajax请求 1.3.3 界面代码优化 二、总线 2.1 定义 2.2 类型分类 2.3 前期准备 2.4 配置组件与路由关系 2.4.1 配置…

Vue Mock.js介绍和使用与首页导航栏左侧菜单搭建

前言&#xff1a; 因为使用Vue开发&#xff0c;组件写的太多&#xff0c;组件与组件之间的传递数据复杂&#xff0c;所以要用到Mock和Bus事件 一&#xff0c;关于Mock 1.1.什么是Mock.js Mock.js是一个模拟数据的生成器&#xff0c;用来帮助前端调试开发、进行前后端的原型分离…

upload-labs靶场未知后缀名解析漏洞

upload-labs靶场未知后缀名解析漏洞 版本影响&#xff1a; phpstudy 版本&#xff1a;5.2.17 ​ 1 环境搭建 1.1 在线靶场下载&#xff0c;解压到phpstudy的www目录下&#xff0c;即可使用 https://github.com/c0ny1/upload-labs1.2 已启动&#xff1a;访问端口9000&…

似然和概率

前言 高斯在处理正态分布的首次提出似然&#xff0c;后来英国物理学家&#xff0c;费歇尔 概率是抛硬币之前&#xff0c;根据环境推断概率 似然则相反&#xff0c;根据结果推论环境 P是关于x的函数&#xff0c;比如x为正面朝上的结果&#xff0c;或者反面朝上的结果&#xf…

数据大爆炸:大数据分析如何改变我们的世界

文章目录 大数据分析的基本概念数据的三个V大数据分析的技术 大数据分析在商业中的应用1. 个性化营销2. 风险管理3. 供应链优化4. 客户服务 大数据分析在医疗保健中的应用1. 疾病预测2. 患者治疗3. 医疗设备监控 大数据分析在科学研究中的应用1. 天文学2. 生物学3. 气象学 大数…

pycharm中配置torch

在控制台cmd中安装好torch后&#xff0c;在pycharm中使用torch&#xff0c;需要进行简单设置即可。 在pycharm中新建一个工程&#xff0c;在file文件中打开setting 在setting中找到project interpreter编译器 找到conda environment的环境配置&#xff0c;设置好相应的目录 新…

2023年蓝帽杯取证复现

案件介绍 2021 年 5 月&#xff0c;公安机关侦破了一起投资理财诈骗类案件&#xff0c;受害人陈昊民向公安机关报案称其在微信上认识一名昵称 为 yang88 的网友&#xff0c;在其诱导下通过一款名为维斯塔斯的 APP &#xff0c;进行投资理财&#xff0c;被诈骗 6 万余万元。接警…

华为OD机考算法题:篮球比赛

目录 题目部分 解读与分析 代码实现 题目部分 题目篮球比赛难度难题目说明篮球(5V5)比赛中&#xff0c;每个球员拥有一个战斗力&#xff0c;每个队伍的所有球员战斗力之和为该队伍的总体战斗力。现有 10 个球员准备分为两队进行训练赛&#xff0c;教练希望 2 个队伍的战斗力…

Java深入理解线程的三大特性

目录 1 CPU缓存导致可见性问题2 线程切换导致原子性问题3 性能优化导致有序性问题4 JMM(Java Memory Model)5 volatile6 synchronized 1 CPU缓存导致可见性问题 线程的三大特性&#xff1a; 可见性&#xff1a;Visibility有序性&#xff1a;Ordering原子性&#xff1a;Atomic…

谈谈最近招人的感受!

最近折腾新的项目&#xff0c;面试了很多实习生小伙伴&#xff0c;我说说我的一些「面试」感受&#xff0c; 虽然是一个老生常谈的话题&#xff0c;但是依然提一下。 准时很重要&#xff1a;提前一点时间&#xff0c;踩个点&#xff0c;别迟到&#xff0c;面试的过程中由于每个…

2023年前端流行什么技术和框架了?

Web前端三大主流框架有React、Vue.js和Angular&#xff0c;由于接触过Vue.js&#xff0c;接下来主讲最新的Vue3.0&#xff01; Vue3.0作为最新版本的Vue.js框架&#xff0c;拥有更强大的性能和更丰富的功能&#xff0c;为低代码开发平台注入了全新的活力。而JNPF快速开发平台作…

浅谈xss

XSS 简介 XSS,全称Cross Site Scripting,即跨站脚本攻击,是最普遍的Web应用安全漏洞。这类漏洞能够使得攻击者嵌入恶意脚本代码到正常用户会访问到的页面中,当正常用户访问该页面时,则可导致嵌入的恶意脚本代码的执行,从而达到恶意攻击用户的目的。需要强调的是,XSS不仅…

使用ElementUI结合Mock完成主页的搭建

目录 一、Mock ( 1 ) 讲述 ( 2 ) 作用 二、引用 三、主页搭建 学习后带来的收获 一、Mock ( 1 ) 讲述 Mock.js是一个用于前端开发中模拟数据的库。它可以帮助开发人员在前端开发过程中模拟接口返回的数据&#xff0c;从而实现前后端分离开发。Mock.js提供了一套简单易…

Python和Scrapy构建可扩展的框架

构建一个可扩展的网络爬虫框架是利用Python和Scrapy实现高效数据采集的重要技能。在本文中&#xff0c;我将为您介绍如何使用Python和Scrapy搭建一个强大灵活的网络爬虫框架。我们将按照以下步骤展开&#xff1a; 1. 安装Scrapy&#xff1a; 首先&#xff0c;确保您已经安装了…

如何计算3种卷积之后的尺寸(普通卷积,转置卷积,空洞卷积)

文章目录 前言一、普通卷积二、转置卷积三、空洞卷积 前言 三种卷积之后的feature map的尺寸如何计算。包括普通卷积&#xff0c;转置卷积&#xff0c;空洞卷积。可以在下面这个链接看到三种卷积的动态图。 卷积动态图 一、普通卷积 普通卷积比较简单了&#xff0c;其计算方式…