一、概述

定义:命令(Command)模式又叫作动作(Action)模式或事务(Transaction)模式,是一种对象的行为模式。将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤消的操作。
上面的定义有以下几个要点:

  • 参数化配置客户的请求:比如对同一个请求者(Invoker),坐下、站起、奔跑三个Command,为Invoker配置不同的的Command,就会执行不同的功能
  • 请求排队:可以将多个命令组成命令队列,然后依次取出命令对象来执行。比如可以按照,坐下-->站起-->奔跑  的队列顺序进行执行
  • 记录请求日志:把请求的历史纪录保存下来,一般是采用永久存储的方式。如果在运行请求的过程中,系统崩溃了,那么当系统再次运行时,就可以从保存的历史记录中获取日志请求,并重新执行命令。
  • 可撤销操作:放弃该操作,回到未执行该操作前的状态。比如原本的状态是 坐下的,执行 站起命令后,想要撤销,再执行 坐下命令。

基本角色:

  • 抽象命令角色:一般定义为接口,用来定义执行命令的接口。
  • 具体命令角色:通常会持有接收者对象,并调用接收者对象的相应功能来完成命令要执行的操作。
  • 接收者角色:真正执行命令的对象。任何类都可能成为接收者,只要它能够实现命令要求实现的相应功能。
  • 调用者角色:要求命令对象执行请求,通常会持有命令对象,可以持有很多的命令对象。这个是客户端真正触发命令并要求命令执行相应操作的地方,也就是说相当于使用命令对象的入口。
  • 客户端角色:创建具体的命令对象,并且设置命令对象的接收者。

二、示例

以游戏玩家玩游戏为例子:

  • 游戏角色(GameRole)作为命令接收者,可以执行坐下,站起来,奔跑三个游戏动作。
  • 将这三个动作封装成4个命令(包括统一的访问接口也就是抽象命令角色(Command),还有就是每一个命令的具体实现SeatCommand、StandUpCommand、RunCommand、StandUpAndRunCommand),最后一个命令是将GameRole的后2个动作封装到了一起。
  • 键盘负责执行具体的命令,在这里作为调用者角色
  • 游戏玩家(Player)作为客户端角色,负责向键盘发送具体的命令

1)命令接收者:真正执行动作的地方,可以是任何执行功能的类

public class GameRole {
public void seat(){
System.out.println("游戏角色打坐");
}
public void standUp(){
System.out.println("游戏角色站起来了");
}
public void run(){
System.out.println("游戏角色奔跑");
}
}

2)抽象命令角色

public interface Command {
void excute();
}

3)具体命令角色

public class SeatCommand implements Command {
private GameRole gameRole;
public SeatCommand(GameRole gameRole){
this.gameRole = gameRole;
}
public void excute() {
gameRole.seat();
}
}
public class StandUpCommand implements Command {
private GameRole gameRole;
public StandUpCommand(GameRole gameRole){
this.gameRole = gameRole;
}
public void excute() {
gameRole.standUp();
}
}
public class RunCommand implements Command {
private GameRole gameRole;
public RunCommand(GameRole gameRole){
this.gameRole = gameRole;
}
public void excute() {
gameRole.run();
}
}
public class StandUpAndRunCommand implements Command {
private GameRole gameRole;
public StandUpAndRunCommand(GameRole gameRole){
this.gameRole = gameRole;
}
public void excute() {
gameRole.standUp();
gameRole.run();
}
}

4)命令调用者

public class Invoker {
private Command command;
public void setCommand(Command command) {
this.command = command;
}
public void invoke(){
command.excute();
} }

5)下发命令的人:游戏玩家

public class Player {
public static void main(String[] args){
GameRole gameRole = new GameRole();
SeatCommand seatCommand = new SeatCommand(gameRole);
StandUpCommand standUpCommand = new StandUpCommand(gameRole);
RunCommand runCommand = new RunCommand(gameRole);
StandUpAndRunCommand standUpAndRunCommand = new StandUpAndRunCommand(gameRole); Invoker invoker = new Invoker();
System.out.println("向角色发起:坐下命令======================");
invoker.setCommand(seatCommand);
invoker.invoke();
System.out.println("向角色发起:站起来命令======================");
invoker.setCommand(standUpCommand);
invoker.invoke();
System.out.println("向角色发起:奔跑命令======================");
invoker.setCommand(runCommand);
invoker.invoke();
System.out.println("向角色发起:坐下命令======================");
invoker.setCommand(seatCommand);
invoker.invoke();
System.out.println("向角色发起:站起来并奔跑命令======================");
invoker.setCommand(standUpAndRunCommand);
invoker.invoke();
}
}
向角色发起:坐下命令======================
游戏角色打坐
向角色发起:站起来命令======================
游戏角色站起来了
向角色发起:奔跑命令======================
游戏角色奔跑
向角色发起:坐下命令======================
游戏角色打坐
向角色发起:站起来并奔跑命令======================
游戏角色站起来了
游戏角色奔跑

更松散的耦合
    命令模式使得发起命令的对象——客户端,和具体实现命令的对象——接收者对象完全解耦,也就是说发起命令的对象完全不知道具体实现对象是谁,也不知道如何实现。
更动态的控制
    命令模式把请求封装起来,可以动态地对它进行参数化、队列化和日志化等操作,从而使得系统更灵活。
很自然的复合命令
    命令模式中的命令对象能够很容易地组合成复合命令,如宏命令,从而使系统操作更简单,功能更强大。
更好的扩展性
    由于发起命令的对象和具体的实现完全解耦,因此扩展新的命令就很容易,只需要实现新的命令对象,然后在装配的时候,把具体的实现对象设置到命令对象中,然后就可以使用这个命令对象,已有的实现完全不用变化。

java常用设计模式十二:命令模式的更多相关文章

  1. java常用设计模式十:模板模式

    一.定义 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中.模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤. 如果上面的话不好理解,请看下面的例子 二.示例 1)定义一个模 ...

  2. java常用设计模式八:代理模式

    一.概述 代理模式是指客户端并不直接调用实际的对象,而是通过调用代理,来间接的调用实际的对象. 其特征是代理类与委托类有同样的接口,真正的核心业务逻辑还是在实际对象里面. 二.为什么要使用代理模式 当 ...

  3. java常用设计模式十一:策略模式

    一.概述 定义:策略模式是指对一系列的算法定义,并将每一个算法封装起来,而且使它们还可以相互替换.策略模式让算法独立于使用它的客户而独立变化.(概念不好理解,可以看第二节的合例子) 基本角色: 环境( ...

  4. java常用设计模式九:桥接模式

    一.概述 将抽象部分与它的实现部分分离,使它们都可以独立地变化.它是一种对象结构型模式.比如存在2个维度,第一个维度有一个抽象类A和对应的子类A1和A2:第二个维度有另一个接口B和对应的子类B1和B2 ...

  5. java常用设计模式三:原型模式

    在说原型模式之前先说一下浅拷贝和深拷贝的概念 一.浅拷贝和深拷贝 1.浅拷贝 在java中,对象创建后需要有一个引用变量来指向该对象实际的地址空间,也就是说引用变量与对象实体是两个不同的数据体.在Ob ...

  6. java常用设计模式五:建造者模式

    1.定义 是一种对象构建的设计模式,它可以将复杂对象的建造过程抽象出来(抽象类别),使这个抽象过程的不同实现方法可以构造出不同表现(属性)的对象. 产品类:一般是一个较为复杂的对象,也就是说创建对象的 ...

  7. java常用设计模式总览

    一.java的设计模式大体上分为三大类: 创建型模式(5种):工厂方法模式,抽象工厂模式,单例模式,建造者模式,原型模式. 结构型模式(7种):适配器模式,装饰器模式,代理模式,外观模式,桥接模式,组 ...

  8. 设计模式(十二)职责链模式(Chain of Responsibility)(对象行为型)

     设计模式(十二)职责链模式(Chain of Responsibility)(对象行为型) 1.概述 你去政府部门求人办事过吗?有时候你会遇到过官员踢球推责,你的问题在我这里能解决就解决,不能解决就 ...

  9. 设计模式 ( 十二 ) 职责链模式(Chain of Responsibility)(对象行为)

     设计模式(十二)职责链模式(Chain of Responsibility)(对象行为型) 1.概述 你去政府部门求人办事过吗?有时候你会遇到过官员踢球推责,你的问题在我这里能解决就解决.不能解决就 ...

随机推荐

  1. 牛客网 Wannafly挑战赛12 删除子串(线性dp)

    题目描述 给你一个长度为n且由a和b组成的字符串,你可以删除其中任意的部分(可以不删),使得删除后的子串“变化”次数小于等于m次且最长. 变化:如果a[i]!=a[i+1]则为一次变化.(且新的字符串 ...

  2. ajax中的contendType和dataType知识点梳理

    在ajax中有2个参数比较重要,之前一直没有搞清楚,下面我们开始梳理一下 1.contentType字段:这个字段的意思,ajax发送给后端的数据是什么类型 如果在ajax中不指定这个属性,则默认是u ...

  3. Java05-Java基础语法(四)循环结构

    Java05-Java基础语法(四)循环结构 循环结构(重复/迭代):根据条件重复执行部分语句 1.while循环结构 while(条件表达式){ 循环体语句; } 1)语法:a.while是关键字 ...

  4. java 向上转型和向下转型

    学习向上转型和向下转型怎么用没多难,但是为什么那样用,我搞了很多次没弄明白.没弄明白的原因是平时学习时之看例子,而例子一般都比较简单,没有对象之间的调用,一般就是一个对象调用自己的方法. 首先看下怎么 ...

  5. jquery ajax 中实现给变量赋值

    我们在用JQuery的Ajax从后台提取数据后想把它赋值给全局变量,但是却怎么都赋不进,为什么呢? 原因其实很简单,我们用的Ajax是异步操作,也就是说在你赋值的时候数据还没提取出来,你当然赋不进去, ...

  6. VS新建API控制器时提示“运行所选代码生成器时出错”

    使用Nuget安装microsoft.entityframeworkcore.tools这个包就行了,安装时注意版本. 根据下图提示应该是新建控制器时用到了这个包,所以安装一下就好了.之前遇到过一次, ...

  7. Mac Terminal

    一.简介   二.实用 1)update-apps-using-terminal-mac https://www.maketecheasier.com/update-apps-using-termin ...

  8. iOS 3D Touch功能

    新的触摸体验——iOS9的3D Touch 一.引言 在iphone6s问世之后,很多果粉都争先要体验3D Touch给用户带来的额外维度上的交互,这个设计之所以叫做3D Touch,其原理上是增加了 ...

  9. Django 访问admin提示ViewDoesNotExist at /admin/

    ViewDoesNotExist at /admin/ Could not import django.views.generic.simple.redirect_to. Parent module ...

  10. LFI/RFI总结

    目录 0×01 文件包含简介 服务器执行PHP文件时,可以通过文件包含函数加载另一个文件中的PHP代码,并且当PHP来执行,这会为开发者节省大量的时间.这意味着您可以创建供所有网页引用的标准页眉或菜单 ...