- 浏览: 103270 次
- 性别:
- 来自: 深圳
文章分类
最新评论
Command(CoR)模式也叫命令模式,是由GoF提出的23种软件设计模式的一种。本文介绍设计模式中的(Command)模式的概念,用法,并用Command模式给出了一个简单的execute/undo实现。
Command模式的概念
Command模式是行为模式之一,Command模式通过被称为Command的类封装了对目标对象的调用行为以及调用参数。Command模式的应用场景
在面向对象的程序设计中,一个对象调用另一个对象,一般情况下的调用过程是:创建目标对象实例;设置调用参数;调用目标对象的方法。但在有些情况下有必要使用一个专门的类对这种调用过程加以封装,我们把这种专门的类称作command类。
- 整个调用过程比较繁杂,或者存在多处这种调用。这时,使用Command类对该调用加以封装,便于功能的再利用。
- 调用前后需要对调用参数进行某些处理。
- 调用前后需要进行某些额外处理,比如日志,缓存,记录历史操作等。
一般来说,Command模式通常可应用到以下场景:
Multi-level undo(多级undo操作)
如果系统需要实现多级回退操作,这时如果所有用户的操作都以command对象的形式实现,系统可以简单地用stack来保存最近执行的命令,如果用户需要执行undo操作,系统只需简单地popup一个最近的command对象然后执行它的undo()方法既可。
Transactional behavior(原子事务行为)
借助command模式,可以简单地实现一个具有原子事务的行为。当一个事务失败时,往往需要回退到执行前的状态,可以借助command对象保存这种状态,简单地处理回退操作。
Progress bars(状态条)
假如系统需要按顺序执行一系列的命令操作,如果每个command对象都提供一个getEstimatedDuration()方法,那么系统可以简单地评估执行状态并显示出合适的状态条。
Wizards(导航)
通常一个使用多个wizard页面来共同完成一个简单动作。一个自然的方法是使用一个command对象来封装wizard过程,该command对象在第一个wizard页面显示时被创建,每个wizard页面接收用户输入并设置到该command对象中,当最后一个wizard页面用户按下 “Finish”按钮时,可以简单地触发一个事件调用execute()方法执行整个动作。通过这种方法,command类不包含任何跟用户界面有关的代码,可以分离用户界面与具体的处理逻辑。
GUI buttons and menu items(GUI按钮与菜单条等等)
Swing系统里,用户可以通过工具条按钮,菜单按钮执行命令,可以用command对象来封装命令的执行。
Thread pools(线程池)
通常一个典型的线程池实现类可能有一个名为addTask()的public方法,用来添加一项工作任务到任务队列中。该任务队列中的所有任务可以用command对象来封装,通常这些command对象会实现一个通用的接口比如java.lang.Runnable。
Macro recording(宏纪录)
可以用command对象来封装用户的一个操作,这样系统可以简单地通过队列保存一系列的command对象的状态就可以记录用户的连续操作。这样通过执行队列中的command对象,就可以完成“Play back”操作了。
Networking
通过网络发送command命令到其他机器上运行。
Parallel Processing(并发处理)
当一个调用共享某个资源并被多个线程并发处理时。
等等
Command模式的结构
Command
Command抽象类。
ConcreteCommand
Command的具体实现类。
Receiver
需要被调用的目标对象。
Invorker
通过Invorker执行Command对象。
Client
调用方。
Command模式的应用范例
下面,我们使用Command模式实现一个简单的execute/undo操作。在该范例中,我们有一个简单的操作:对字符串做append操作,这个操作由Receiver类实现;另外,我们需要记录操作历史,并能简单加以回退(undo),所以我们采用Command模式实现。
文件一览:
Client
测试类
Command
Command抽象类
UndoableCommand
支持undo操作的Command抽象类,该类是Command类的子类
ConcreteCommand
具体的UndoableCommand实现类,该类继承UndoableCommand类,所以支持undo操作
CommandManager
Command管理类。该类使用Stack来管理执行过的Command对象,并提供executeCommand()与undoCommand()方法
Receiver
执行任务的目标类
Invoker
这个类在我们的例里没有被用到,但我们仍给出了它的一个参考实现,不过用注释表示它不可用
代码:
import java.util.Stack;
public class Client {
/**
* Test Command Pattern
*
*/
public static void main(String[] args) {
CommandManager commandMgr = new CommandManager();
Receiver receiver = new Receiver();
System.out.println("--- execute command ---");
Command commandAaa = new ConcreteCommand(receiver, "aaa");
commandMgr.executeCommand(commandAaa);
Command commandBbb = new ConcreteCommand(receiver, "bbb");
commandMgr.executeCommand(commandBbb);
Command commandCcc = new ConcreteCommand(receiver, "ccc");
commandMgr.executeCommand(commandCcc);
Command commandDdd = new ConcreteCommand(receiver, "ddd");
commandMgr.executeCommand(commandDdd);
System.out.println(receiver.getData());
System.out.println("-- undo ---");
commandMgr.undoCommand();
commandMgr.undoCommand();
System.out.println(receiver.getData());
}
}
/**
* Command
* abstract command class
*
*/
abstract class Command {
protected Receiver receiver;
protected String param;
public Command(Receiver receiver, String expr) {
this.receiver = receiver;
this.param = expr;
}
abstract public void execute();
}
/**
* UndoableCommand
* abstract undo supportable command class which extends from Command class
*
*/
abstract class UndoableCommand extends Command {
public UndoableCommand(Receiver receiver, String expr) {
super(receiver, expr);
}
abstract public void undo();
}
/**
* ConcreteCommand
* concrete command class which extends from UndoableCommand
*
*/
class ConcreteCommand extends UndoableCommand {
private String previousData = null;
public ConcreteCommand(Receiver receiver, String expr) {
super(receiver, expr);
}
@Override
public void execute() {
previousData = receiver.getData();
receiver.append(this.param);
}
@Override
public void undo() {
receiver.setData(previousData);
}
}
/**
* CommandManager
* Command Manager class which stack the exe
*
*/
class CommandManager {
private Stack commandStack = new Stack();
public void executeCommand(Command cmd) {
cmd.execute();
if (cmd instanceof UndoableCommand) {
commandStack.push(cmd);
}
}
public void undoCommand() {
if (commandStack.size() > 0) {
UndoableCommand cmd = (UndoableCommand) commandStack.pop();
cmd.undo();
} else {
throw new UnsupportedOperationException("");
}
}
}
/**
* Receiver
* target object
*
*/
class Receiver {
private String data = "";
public void append(String expr) {
data += expr;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
/**
class Invoker {
private Command command;
public void setCommand(Command command) {
this.command = command;
}
public void executeCommand() {
command.execute();
}
}
*/
public class Client {
/**
* Test Command Pattern
*
*/
public static void main(String[] args) {
CommandManager commandMgr = new CommandManager();
Receiver receiver = new Receiver();
System.out.println("--- execute command ---");
Command commandAaa = new ConcreteCommand(receiver, "aaa");
commandMgr.executeCommand(commandAaa);
Command commandBbb = new ConcreteCommand(receiver, "bbb");
commandMgr.executeCommand(commandBbb);
Command commandCcc = new ConcreteCommand(receiver, "ccc");
commandMgr.executeCommand(commandCcc);
Command commandDdd = new ConcreteCommand(receiver, "ddd");
commandMgr.executeCommand(commandDdd);
System.out.println(receiver.getData());
System.out.println("-- undo ---");
commandMgr.undoCommand();
commandMgr.undoCommand();
System.out.println(receiver.getData());
}
}
/**
* Command
* abstract command class
*
*/
abstract class Command {
protected Receiver receiver;
protected String param;
public Command(Receiver receiver, String expr) {
this.receiver = receiver;
this.param = expr;
}
abstract public void execute();
}
/**
* UndoableCommand
* abstract undo supportable command class which extends from Command class
*
*/
abstract class UndoableCommand extends Command {
public UndoableCommand(Receiver receiver, String expr) {
super(receiver, expr);
}
abstract public void undo();
}
/**
* ConcreteCommand
* concrete command class which extends from UndoableCommand
*
*/
class ConcreteCommand extends UndoableCommand {
private String previousData = null;
public ConcreteCommand(Receiver receiver, String expr) {
super(receiver, expr);
}
@Override
public void execute() {
previousData = receiver.getData();
receiver.append(this.param);
}
@Override
public void undo() {
receiver.setData(previousData);
}
}
/**
* CommandManager
* Command Manager class which stack the exe
*
*/
class CommandManager {
private Stack commandStack = new Stack();
public void executeCommand(Command cmd) {
cmd.execute();
if (cmd instanceof UndoableCommand) {
commandStack.push(cmd);
}
}
public void undoCommand() {
if (commandStack.size() > 0) {
UndoableCommand cmd = (UndoableCommand) commandStack.pop();
cmd.undo();
} else {
throw new UnsupportedOperationException("");
}
}
}
/**
* Receiver
* target object
*
*/
class Receiver {
private String data = "";
public void append(String expr) {
data += expr;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
/**
class Invoker {
private Command command;
public void setCommand(Command command) {
this.command = command;
}
public void executeCommand() {
command.execute();
}
}
*/
执行Client,输出结果:
C:\Command>javac *.java
C:\Command>java Client
--- execute command ---
aaabbbcccddd
-- undo ---
aaabbb
C:\Command>
我们可以看到,使用Command模式非常简单地就实现了undo操作。C:\Command>java Client
--- execute command ---
aaabbbcccddd
-- undo ---
aaabbb
C:\Command>
发表评论
-
设计模式之 State - 状态模式
2010-12-18 16:49 1122State模式也叫状态模式,是由GoF提出的23种软件设计模式 ... -
设计模式之 Strategy - 策略模式
2010-12-18 16:46 1390Strategy模式也叫策略模 ... -
设计模式之 Visitor - 访问者模式
2010-12-18 16:45 1484Visitor模式也叫访问者 ... -
设计模式之 Template Method - 模板方法模式
2010-12-18 16:41 1601Template Method模式也叫模板方法模式,是由G ... -
设计模式之 Observer - 观察者模式
2010-12-18 16:33 1437Observer模式也叫观察者 ... -
设计模式之 Memento - 备忘录模式
2010-12-18 16:31 1093Memento模式也叫备忘录 ... -
设计模式之 Mediator - 中介者模式
2010-12-18 16:29 1150Mediator模式也叫中介者模式,是由GoF提出的23种软件 ... -
设计模式之 Iterator - 迭代模式
2010-12-18 16:27 1201Iterator模式也叫迭代模式,是由GoF提出的23种设 ... -
设计模式之 Interpreter - 解释器模式
2010-12-18 16:23 1032Interpreter模式也叫解释器模式,是由GoF提出的 ... -
设计模式之 Chain of Responsibility - 职责链模式
2010-12-18 16:18 1236Chain of Responsibility(CoR)模 ... -
设计模式之 Proxy - 代理模式
2010-12-16 23:45 1090Proxy模式也叫代理模式,是由GoF提出的23种软件设计 ... -
设计模式之 Flyweight - 享元模式
2010-12-16 23:42 1085Flyweight模式也叫享元模式,是由GoF提出的23种 ... -
设计模式之 Facade - 外观模式
2010-12-16 23:41 1044Facade模式也叫外观模式 ... -
设计模式之 Singleton - 单态模式
2010-11-26 11:58 887Singleton模式也叫单态模式,是由GoF提出的2 ... -
设计模式之 Decorator - 装饰模式
2010-11-25 23:48 838装饰模式,The Decorator Pattern atta ... -
设计模式之 Composite - 组合模式
2010-11-25 23:35 1494Composite模式也叫组合 ... -
设计模式之 Bridge - 桥接模式
2010-11-25 23:08 1045Bridge模式也叫桥接模式,是由GoF提出的23种软件设计模 ... -
设计模式之 Adapter - 适配器模式
2010-11-25 22:58 952Adapter模式也叫适配器模式,是由GoF提出的23种设 ... -
设计模式之Prototype - 原型模式
2010-11-25 22:14 1165Prototype模式也叫原型模式,是由GoF提出的23种设计 ... -
设计模式之 Builder - 建造者模式
2010-11-18 23:51 1100Builder模式也叫建造者模式或者生成器模式,是由GoF ...
相关推荐
自动丝印设备(sw18可编辑+工程图+Bom)全套设计资料100%好用.zip
链板式连续提升机6米高度(sw18可编辑+工程图)全套设计资料100%好用.zip
万能拍照识物是一款多功能拍照识别应用,可以拍照识别动物、植物、菜肴、车型、二维码和条形码、物品等。支持从手机内存选取图片识别,识别文字后可一键复制,显示识别结果后可一键搜索更多资料。操作方便,识别准确。 【使用方法】:安装软件后,打开相机对准想要识别的对象进行拍照,或者从相册中选择图片进行识别。识别完成后,可以选择复制文字或搜索更多信息。
本项目是一个基于Spring Boot的老年一站式服务平台,旨在为老年人提供一个综合性的服务体验。平台集成了健康管理、社交联系、生活服务等多项功能,通过简洁直观的用户界面,使老年人能够轻松享受到便捷的服务。在健康管理方面,平台提供了定期体检提醒、用药记录跟踪等功能,帮助老年人更好地管理自身健康。社交联系功能则允许老年人与家人、朋友以及社区的其他成员保持联系,减少孤独感。生活服务功能涵盖了日常购物、家政预约等方面,极大地便利了老年人的日常生活。本项目采用Spring Boot框架进行开发,确保了系统的稳定性和易扩展性。通过本项目的开发,旨在提升老年人的生活质量,促进他们的身心健康,构建一个更加友好和谐的生活环境。 项目为完整毕设源码,先看项目演示,希望对需要的同学有帮助。
本项目是一个基于Java的Spring Boot框架开发的毕业设计论文期末大作业,旨在构建一个教学辅助平台。该项目主要功能包括课程资源管理、在线学习、作业提交与批改、学生与教师互动交流等。系统采用分层架构设计,核心业务逻辑由服务层处理,数据访问由数据访问层实现,前端界面友好,用户体验良好。通过Spring Boot框架,项目实现了快速开发和部署,利用其内置的Tomcat服务器和自动配置特性,简化了开发流程。数据库采用MySQL存储系统数据,确保数据的稳定性和安全性。项目的开发不仅展示了学生对Java和Spring Boot技术的掌握,还体现了其在教学辅助系统设计方面的能力,适应未来教育信息化的趋势。项目为完整毕设源码,先看项目演示,希望对需要的同学有帮助。
本项目是一个基于Spring Boot框架的在线家具商城设计与实现,旨在利用现代信息技术为学生提供一个完整的毕业设计实践平台。系统主要功能涵盖用户注册与登录、商品展示与搜索、购物车管理、订单处理、支付接口集成以及用户评价系统等。通过使用Spring Boot框架,项目实现了前后端分离,前端采用Vue.js技术栈,后端则利用Spring Security进行安全管理,确保用户数据的安全性。数据库设计采用MySQL,支持商品信息的存储和管理。项目的开发不仅帮助学生理解电子商务系统的基本架构和运作流程,还涉及到了现代Web开发中常用的多种技术和工具,如RESTful API设计、JWT认证机制等。通过此项目,学生可以实践从系统需求分析、设计、编码到测试的完整软件开发周期。 项目为完整毕设源码,先看项目演示,希望对需要的同学有帮助。
功能说明: 生鲜食品o2o商城系统,主要的模块包括查看管理员;首页、个人中心、用户管理、商品分类管理、商品细类管理、商品类型管理、商品信息管理、折扣商品管理、新品上市管理、商品评价管理、系统管理、订单管理等等。前台;首页、商品信息、折扣商品、新品上市、新闻资讯、我的个人中心、后台管理、购物车、客服等等。用户后台:个人中心、个人信息、商品信息管理、商品评价管理、我的收藏管理、订单管理等等。环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7及以上 数据库工具:Navicat11及以上 开发软件:eclipse/idea Maven包:Maven3.3及以上 服务器:tomcat7及以上。。。
selenium+google浏览器
项目资源包含:可运行源码+sql文件+ 适用人群:学习不同技术领域的小白或进阶学习者;可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。 项目具有较高的学习借鉴价值,也可拿来修改、二次开发。 有任何使用上的问题,欢迎随时与博主沟通,博主看到后会第一时间及时解答。 开发语言:Java 框架:springboot 技术:Vue JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7(一定要5.7版本) 数据库工具:Navicat11 开发软件:eclipse/myeclipse/idea Maven包:Maven3.3.9 系统是一个很好的项目,结合了后端服务(SpringBoot)和前端用户界面(Vue.js)技术,实现了前后端分离。
数字信号处理的设计 关于信号的谱分析 主要用到DFT与FFT的实现与比较 以及窗函数的应用
本项目是一个基于Spring Boot框架的经方药食两用服务平台,旨在利用现代信息技术提升中医药服务的效率和质量。该项目主要功能包括中医药方剂的查询与管理、食材与药食同源物品的信息检索、个性化健康建议的生成以及用户管理模块。通过Spring Boot框架,项目实现了快速开发和部署,提供了稳定的性能和良好的扩展性。 在技术实现上,项目采用了前后端分离的架构,前端使用Vue.js进行开发,后端则利用Spring Boot框架处理业务逻辑和数据交互。数据库选用MySQL,确保数据的持久化和高效查询。此外,项目还集成了Spring Security进行安全管理,保障用户数据的安全。 该项目不仅为中医药爱好者提供了一个便捷的信息查询平台,还为健康管理专业人士提供了一个辅助工具,助力中医药的现代化和普及化。项目为完整毕设源码,先看项目演示,希望对需要的同学有帮助。
无人机频射信号检测数据集,平均正确识别率在94.3%,支持yolov11格式的标注,364张原始图片
PaddleOCR是一款由阿里云开发的开源OCR(Optical Character Recognition,光学字符识别)系统,其主要目标是解决图像中的文字识别问题。该系统以其高效、准确和灵活的特点,在业界得到了广泛应用。在“paddleOCR文字识别,只需要ONNX,支持C++,PYTHON版本”的标题和描述中,我们可以提取出以下几个关键知识点: 1. **PaddleOCR**: PaddleOCR是一个基于PaddlePaddle深度学习框架的轻量级OCR工具包。它提供了丰富的预训练模型,包括用于检测的DB和用于识别的CRNN等,能够处理多种语言和场景的文字识别任务。 2. **ONNX(Open Neural Network Exchange)**: ONNX是一种开放的模型交换格式,允许开发者在不同的深度学习框架之间转换模型,如从PaddlePaddle转换到TensorFlow或PyTorch。在PaddleOCR中,支持ONNX意味着用户可以将训练好的PaddleOCR模型导出为ONNX格式,然后在C++或Python等不同环境中运行,实现跨平台的推理。 3. **C+
1. 设置配置文件中的微信公众测试号的appid和secret,以及接收者的openId ```properties wx: appid= secret= openId= weather: #女朋友的生日 格式 yyyy-mm-dd birthday= #情窦初开的日子 格式 yyyy-mm-dd love= #生日按照哪种日历 true阴历,false洋历 chineseDate=true #消息模板ID templateId= ``` 2. 模板内容信息如下 ```text 今天是我们恋爱的第{{love.DATA}}天 距离乖乖宝贝的生日还有{{birthday.DATA}}天 乖乖宝贝的城市:{{city.DATA}} 今天的日期是:{{date.DATA}} 下面由我来给你播报一下天气状况 (o‘ .) 今天的天气:{{weather.DATA}} 最低气温:{{minTemperature.DATA}} 最高气温:{{maxTemperature.DATA}} 湿度:{{humidity.DATA}} 风向:{{windDir
abaqus子弹穿钢板模型,cae文件
项目资源包含:可运行源码+sql文件+ 适用人群:学习不同技术领域的小白或进阶学习者;可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。 项目具有较高的学习借鉴价值,也可拿来修改、二次开发。 有任何使用上的问题,欢迎随时与博主沟通,博主看到后会第一时间及时解答。 开发语言:Java 框架:springboot 技术:Vue JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7(一定要5.7版本) 数据库工具:Navicat11 开发软件:eclipse/myeclipse/idea Maven包:Maven3.3.9 系统是一个很好的项目,结合了后端服务(SpringBoot)和前端用户界面(Vue.js)技术,实现了前后端分离。
功能说明: 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7及以上 数据库工具:Navicat11及以上 开发软件:eclipse/idea Maven包:Maven3.3及以上 服务器:tomcat7及以上。。。
均值移动聚类算法演示 参考文献:Dorin Comaniciu 和 Peter Meer,"均值移动: 特征空间分析的稳健方法"。 IEEE Transactions on Pattern Analysis and Machine Intelligence. 2002. pp.
功能说明: 奥博羽毛球俱乐部管理系统为用户提供羽毛球俱乐部场地预约服务,系统包括前台和后台。 前台用户可以先进行注册,填写个人的基本信息提交到服务器,服务器把数据保存到数据库。管理员对会员的信息进行验证后,会员通过验证后的用户名和密码进行登录,登录之后查看教练信息、场地信息、教学视频、公告资讯、意见建议。会员可以查看教练信息、场地信息,可以进行预约,填写预约信息后,生成预约订单。点开我的功能模块,修改个人信息,重新登录等功能。 管理员是后台的管理者,对用户提交的预约订单进行审核,如果不是正常的预定或者违规的内容进行删除,也可以要求会员进行修改后重新提交。管理员对教练信息、场地信息、教学视频、意见建议进行管理,管理员还可以对用户信息进行管理,发布公告推送给用户端。环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7及以上 数据库工具:Navicat11及以上 开发软件:eclipse/idea Maven包:Maven3.3及以上 服务器:tomcat7及以上。。。
SpringBoot与Shiro的整合是Java Web开发中常见的安全控制技术组合,主要用于实现用户认证、授权、会话管理和安全管理等功能。下面将详细讲解这个完整代码案例的实现过程和相关知识点。 1. **Spring Boot简介** Spring Boot是Spring框架的一个扩展,旨在简化Spring应用的初始搭建以及开发过程。它内置了Tomcat服务器,可以快速构建独立运行的Spring应用程序,并提供了自动配置功能,大大减少了开发者的工作量。 2. **Shiro简介** Apache Shiro是一款轻量级的安全框架,提供认证、授权、会话管理及加密等核心安全服务。它的设计目标是简单易用,让开发者能够快速地在应用中集成安全控制。 3. **整合步骤** - **添加依赖**:在Spring Boot的`pom.xml`文件中,我们需要引入Spring Boot的web依赖和Shiro的相关依赖,如`spring-boot-starter-web`、`shiro-spring-boot-starter`等。 - **配置Shiro**:创建一个`ShiroConfig`