深入Java设计模式之命令模式

简介: 深入Java设计模式之命令模式

概念

意图:将一个请求封装成一个对象,从而使您可以用不同的请求对客户进行参数化。

主要解决:在软件系统中,行为请求者与行为实现者通常是一种紧耦合的关系,但某些场合,比如需要对行为进行记录、撤销或重做、事务等处理时,这种无法抵御变化的紧耦合的设计就不太合适。

何时使用:在某些场合,比如要对行为进行"记录、撤销/重做、事务"等处理,这种无法抵御变化的紧耦合是不合适的。在这种情况下,如何将"行为请求者"与"行为实现者"解耦?将一组行为抽象为对象,可以实现二者之间的松耦合。

如何解决:通过调用者调用接受者执行命令,顺序:调用者→接受者→命令。

 

命令模式调用顺序

代码规范

package com.lzhsite.technology.designPattern.command.GeneralDemo;
/**
 * 系统需要将请求调用者和请求接收者解耦,使得调用者和接收者不直接交互。
 * @author lzhcode
 *
 */
public class ClientForGeneralDemo
{
    public static void main(String[] args)
    {
        Receiver receiver = new Receiver();
        Command command = new ConcreteCommand(receiver);
        Invoker invoker = new Invoker();
        invoker.setCommand(command);
    }
}
package com.lzhsite.technology.designPattern.command.GeneralDemo;
public interface Command
{
    public void execute();
}
package com.lzhsite.technology.designPattern.command.GeneralDemo;
public class ConcreteCommand implements Command
{
    private Receiver receiver = null;
    private String state;
    public ConcreteCommand(Receiver receiver)
    {
        this.receiver = receiver;
    }
    public void execute()
    {
        receiver.action();
    }
}
package com.lzhsite.technology.designPattern.command.GeneralDemo;
public class Invoker
{
    private Command command = null;
    public void setCommand(Command command)
    {
        this.command = command;
    }
    public void runCommand()
    {
        command.execute();
    }
}
package com.lzhsite.technology.designPattern.command.GeneralDemo;
public class Receiver
{
    public void action()
    {
    }
}

适用场景

1.系统需要将请求调用者和请求接收者解耦,使得调用者和接收者不直接交互。

2.系统需要在不同的时间指定请求、将请求排队和执行请求。

3.系统需要支持命令的撤销(Undo)操作和恢复(Redo)操作。

4.系统需要将一组操作组合在一起,即支持宏命令。

优点

1.降低对象之间的耦合度

2.新的命令可以很容易地加入到系统中。

3.可以比较容易地设计一个组合命令。

4.调用同一方法实现不同的功能

缺点

使用命令模式可能会导致某些系统有过多的具体命令类。因为针对每一个命令都需要设计一个具体命令类,因此某些系统可能需要大量具体命令类,这将影响命令模式的使用。

实际应用

点餐问题

定义命令接口

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public interface ICommand
{
    public void execute();
    public void setCook(ICook cook);
    public int getTableNumber();
}

某道菜命令

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class PorkCommand implements ICommand
{
    private int tableNumber;
    private ICook cook = null;
    public PorkCommand(int tableNumber)
    {
        this.tableNumber = tableNumber;
    }
    public void setCook(ICook cook)
    {
        this.cook = cook;
    }
    public int getTableNumber()
    {
        return this.tableNumber;
    }
    public void execute()
    {
        this.cook.cook(tableNumber, "蒜泥白肉");
    }
}
package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class ChopCommand implements ICommand {
  private int tableNumber;
  private ICook cook = null;
  public ChopCommand(int tableNumber) {
    this.tableNumber = tableNumber;
  }
  public void setCook(ICook cook) {
    this.cook = cook;
  }
  public int getTableNumber() {
    return this.tableNumber;
  }
  public void execute() {
    this.cook.cook(tableNumber, "绿豆排骨煲");
  }
}
package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class DuckCommand implements ICommand
{
    private int tableNumber;
    private ICook cook = null;
    public DuckCommand(int tableNumber)
    {
        this.tableNumber = tableNumber;
    }
    public void setCook(ICook cook)
    {
        this.cook = cook;
    }
    public int getTableNumber()
    {
        return this.tableNumber;
    }
    public void execute()
    {
        this.cook.cook(tableNumber, "北京烤鸭");
    }
}

点菜命令队列

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
import java.util.Queue;
import java.util.concurrent.ArrayBlockingQueue;
/**
 * 订单列表
 * 
 * @author lzhcode
 *
 */
public class CommandQueue {
  private static Queue<ICommand> commands = new ArrayBlockingQueue<ICommand>(200);
  public  static void addMenu(MenuCommand menuCommand) {
    for (ICommand command : menuCommand.getCommands()) {
      commands.add(command);
    }
  }
  public static void removeOneCommand() {
    if (commands.size() > 0) {
      commands.poll();
    }
  }
  public static ICommand getOneCommand() {
    ICommand command = null;
    if (commands.size() > 0) {
      command = commands.peek();
    }
    return command;
  }
}

订单命令

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
import java.util.Collection;
import java.util.concurrent.ArrayBlockingQueue;
public class MenuCommand implements ICommand
{
    private Collection<ICommand> commands = new ArrayBlockingQueue<ICommand>(100);
    public void addCommand(ICommand command)
    {
        commands.add(command);
    }
    public void execute()
    {
        CommandQueue.addMenu(this);
    }
    public void setCook(ICook cook)
    {
    }
    public int getTableNumber()
    {
        return 0;
    }
    public Collection<ICommand> getCommands()
    {
        return this.commands;
    }
}

定义菜类型接口

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public interface ICook
{
    public void cook(int tableNumber, String name);
}
package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class HotCook implements ICook, Runnable {
  private String name;
  public HotCook(String name) {
    this.name = name;
  }
  public void cook(int tableNumber, String name) {
    int cookTime = (int) (20 * Math.random());
    System.out.println(this.name + " 热菜厨师正在为 " + tableNumber + " 号桌做:" + name);
    try {
      Thread.sleep(cookTime);
    } catch (InterruptedException exception) {
      exception.printStackTrace();
    }
    System.out.println(this.name + " 热菜厨师为 " + tableNumber + " 号桌做好了:" + name + ", 共计耗时 = " + cookTime + " 秒");
  }
  public void run() {
    while (true) {
      synchronized (ClientForRestaurantDemo.Lock) {
        ICommand command = CommandQueue.getOneCommand();
        if (command != null) {
          if(command instanceof DuckCommand || command instanceof ChopCommand){
            command.setCook(this);
            command.execute();
            CommandQueue.removeOneCommand();
          }
        }
      }
      try {
        Thread.sleep(1000l);
      } catch (InterruptedException exception) {
        exception.printStackTrace();
      }
    }
  }
}
package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class ColdCook implements ICook, Runnable {
  private String name;
  public ColdCook(String name) {
    this.name = name;
  }
  public void cook(int tableNumber, String name) {
    int cookTime = (int) (20 * Math.random());
    System.out.println(this.name + " 冷菜厨师正在为 " + tableNumber + " 号桌做:" + name);
    try {
      Thread.sleep(cookTime);
    } catch (InterruptedException exception) {
      exception.printStackTrace();
    }
    System.out.println(this.name + " 冷菜厨师为 " + tableNumber + " 号桌做好了:" + name + ", 共计耗时 = " + cookTime + " 秒");
  }
  public void run() {
    while (true) {
      synchronized (ClientForRestaurantDemo.Lock) {
        ICommand command = CommandQueue.getOneCommand();
        if (command != null) {
          if(command instanceof PorkCommand){
            command.setCook(this);
            command.execute();
            CommandQueue.removeOneCommand();
          }
        }
      }
      try {
        Thread.sleep(1000l);
      } catch (InterruptedException exception) {
        exception.printStackTrace();
      }
    }
  }
}

定义服务员

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class Waiter
{
    public MenuCommand menuCommand = new MenuCommand();
    public void orderDish(ICommand command)
    {
        menuCommand.addCommand(command);
    }
    public void orderOver()
    {
        this.menuCommand.execute();
    }
}

程序启动入口

package com.lzhsite.technology.designPattern.command.RestaurantDemo;
import java.util.concurrent.locks.Lock;
/**
 * 系统需要在不同的时间指定请求、将请求排队和执行请求。 系统需要将一组操作组合在一起,即支持宏命令。
 * 
 * @author lzhcode
 *
 */
public class ClientForRestaurantDemo {
  public static Byte Lock = 1;
  public static void main(String[] args) {
    // 4个厨师同时开始工作
    CookManager.runCookManager();
    // 一次循环相当于一个订单
    for (int i = 0; i < 5; i++) {
      Waiter waiter = new Waiter();
      ICommand chopCommand = new ChopCommand(i);
      ICommand duckCommand = new DuckCommand(i);
      ICommand porkCommand = new PorkCommand(i);
      waiter.orderDish(chopCommand);
      waiter.orderDish(duckCommand);
      waiter.orderDish(porkCommand);
      waiter.orderOver();
    }
  }
}
package com.lzhsite.technology.designPattern.command.RestaurantDemo;
public class CookManager {
  public static void runCookManager() {
    HotCook cook1 = new HotCook("张三");
    HotCook cook2 = new HotCook("李四");
    ColdCook cook3 = new ColdCook("王五");
    ColdCook cook4 = new ColdCook("钱六");
    Thread thread1 = new Thread(cook1);
    thread1.start();
    Thread thread2 = new Thread(cook2);
    thread2.start();
    Thread thread3 = new Thread(cook3);
    thread3.start();
    Thread thread4 = new Thread(cook4);
    thread4.start();
  }
}

代码附录

https://gitee.com/lzhcode/maven-parent/tree/master/lzh-technology/src/main/java/com/lzhsite/technology/designPattern/command


目录
相关文章
|
29天前
|
设计模式 Java Spring
Java 设计模式之责任链模式:优雅处理请求的艺术
责任链模式通过构建处理者链,使请求沿链传递直至被处理,实现发送者与接收者的解耦。适用于审批流程、日志处理等多级处理场景,提升系统灵活性与可扩展性。
202 2
|
29天前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
248 0
|
3月前
|
设计模式 缓存 Java
Java设计模式(二):观察者模式与装饰器模式
本文深入讲解观察者模式与装饰器模式的核心概念及实现方式,涵盖从基础理论到实战应用的全面内容。观察者模式实现对象间松耦合通信,适用于事件通知机制;装饰器模式通过组合方式动态扩展对象功能,避免子类爆炸。文章通过Java示例展示两者在GUI、IO流、Web中间件等场景的应用,并提供常见陷阱与面试高频问题解析,助你写出灵活、可维护的代码。
|
28天前
|
设计模式 算法 搜索推荐
Java 设计模式之策略模式:灵活切换算法的艺术
策略模式通过封装不同算法并实现灵活切换,将算法与使用解耦。以支付为例,微信、支付宝等支付方式作为独立策略,购物车根据选择调用对应支付逻辑,提升代码可维护性与扩展性,避免冗长条件判断,符合开闭原则。
244 35
|
28天前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
209 8
|
6月前
|
设计模式 缓存 安全
【高薪程序员必看】万字长文拆解Java并发编程!(8):设计模式-享元模式设计指南
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。
158 0
|
3月前
|
设计模式 安全 Java
Java设计模式(一):单例模式与工厂模式
本文详解单例模式与工厂模式的核心实现及应用,涵盖饿汉式、懒汉式、双重检查锁、工厂方法、抽象工厂等设计模式,并结合数据库连接池与支付系统实战案例,助你掌握设计模式精髓,提升代码专业性与可维护性。
|
3月前
|
设计模式 XML 安全
Java枚举(Enum)与设计模式应用
Java枚举不仅是类型安全的常量,还具备面向对象能力,可添加属性与方法,实现接口。通过枚举能优雅实现单例、策略、状态等设计模式,具备线程安全、序列化安全等特性,是编写高效、安全代码的利器。
|
6月前
|
设计模式 消息中间件 Java
【设计模式】【行为型模式】命令模式(Command)
一、入门 什么是命令模式? 命令模式是一种行为设计模式,它将请求或操作封装为对象,从而使你可以用不同的请求对客户进行参数化,并支持请求的排队、记录、撤销等操作。 命令模式的核心是将“请求”封装为独立的
219 15
|
9月前
|
设计模式 存储 算法
「全网最细 + 实战源码案例」设计模式——命令模式
命令模式(Command Pattern)是一种行为型设计模式,将请求封装成独立对象,从而解耦请求方与接收方。其核心结构包括:Command(命令接口)、ConcreteCommand(具体命令)、Receiver(接收者)和Invoker(调用者)。通过这种方式,命令的执行、撤销、排队等操作更易扩展和灵活。 适用场景: 1. 参数化对象以操作。 2. 操作放入队列或远程执行。 3. 实现回滚功能。 4. 解耦调用者与接收者。 优点: - 遵循单一职责和开闭原则。 - 支持命令组合和延迟执行。 - 可实现撤销、恢复功能。 缺点: - 增加复杂性和类数量。
299 14
「全网最细 + 实战源码案例」设计模式——命令模式