Java设计模式 - 命令模式
😄生命不息,写作不止
🔥 继续踏上学习之路,学之分享笔记
👊 总有一天我也能像各位大佬一样
🏆 一个有梦有戏的人 @怒放吧德德
🌝分享学习心得,欢迎指正,大家一起学习成长!
简介
命令模式(Command Pattern)是一种数据驱动的设计模式,它属于行为型模式。请求以命令的形式包裹在对象中,并传给调用对象。调用对象寻找可以处理该命令的合适的对象,并把该命令传给相应的对象,该对象执行命令。
———— 菜鸟联盟
命令模式解析
命令模式会将一个请求封装为一个接口对象,由各种功能去实现其方法,在详细命令类中,通过聚合的方式将具体方法进行抽取调用。
UML图:
命令模式角色和职责
1)、Invoker:调用者角色
2)、Command:命令角色,需要执行的所有命令,(抽象类或接口)
3)、Receiver:命令接受者,知道如何实施执行某个命令
4)、ConcreteCommand:将一个接受对象和一个命令相互绑定,调用具体应操作的方法。
命令模式实例
通过智能家居的例子来学习命令模式,有一个遥控器,上面集成了好多台设备的开关操作,有灯光、电视等控制按钮。采用命令模式来管理控制,具体以下一步一步解析。首先看看具体的类图:
本次的类结构看起来比较复杂,其实原理都是Java面向对象。
本次实验通过定义命令接口,里面包含执行的命令,在通过各种各样设备去实现命令接口及其方法,不同设备的不同操作,如开、关,各种视为一个类,在这个类中实现不同的操作方法,虽然命令不同,但是可以通过聚合的方式,使用命令接受者类中去实现具体的不同方法进行分别调用。最后定义遥控器定义命令组并且为按钮去绑定相应的实现命令。
代码如下,会一步一步解释。
①、定义命令角色
命令角色可以是接口也可以是抽象类,根据本次的案例设计为接口。
只有一个执行方法。
package com.lyd.demo.command;
/**
* @Author: lyd
* @Description: 接口 - 命令
* @Date: 2022-09-03
*/
public interface Command {
public void execute(); // 执行方法
}
②、定义空命令
定义空命令是为了防止判空
package com.lyd.demo.command;
/**
* @Author: lyd
* @Description: 空命令方法,简化空判断
* @Date: 2022-09-03
*/
public class NoCommand implements Command {
@Override
public void execute() {
}
}
③、定义接受者
接收者就是具体的实现方法。
package com.lyd.demo.command.light;
/**
* @Author: lyd
* @Description: 被聚合类,也就是真正细节执行方法
* @Date: 2022-09-03
*/
public class LightReceive {
public void on() {
System.out.println(" 灯光已打开... ");
}
public void off() {
System.out.println(" 灯光已关闭... ");
}
}
④、定义实现类
灯光打开类,通过聚合方式获取具体实现方法;灯光实现类实现命令接口,实现执行方法(调用聚合类中的打开方法)
package com.lyd.demo.command.light;
import com.lyd.demo.command.Command;
/**
* @Author: lyd
* @Description: 实现类 - 点灯打开类
* @Date: 2022-09-03
*/
public class LightOnCommand implements Command {
// 聚合
LightReceive lightReceive;
public LightOnCommand(LightReceive lightReceive) {
this.lightReceive = lightReceive;
}
@Override
public void execute() { // 灯光打开命令只需要调用灯光开启的方法
lightReceive.on();
}
}
关闭命令也是如此设置,这里不粘贴代码了
⑤、定义遥控器
因为遥控器有许多设备的开关,定义相应开关按钮的命令组,通过构造方法去初始化,并把美格尔对象实例化为空命令对象。通过setCommand方法来绑定按钮和命令,onButton是模拟按下开关时候调用的方法。
package com.lyd.demo.controller;
import com.lyd.demo.command.Command;
import com.lyd.demo.command.NoCommand;
/**
* @Author: lyd
* @Description: 遥控器
* @Date: 2022-09-03
*/
public class RemoteController {
// 开按钮的命令组
Command[] onCommand;
// 关闭命令组
Command[] offCommand;
public RemoteController() { // 假设有舞台设备,每台设备都是开关命令
onCommand = new Command[5];
offCommand = new Command[5];
// 初始化
for (int i=0; i<5; i++) {
onCommand[i] = new NoCommand();
offCommand[i] = new NoCommand();
}
}
/**
* 给按钮设置命令
* @param no 编号-代表设备
* @param onCommand - 开命令
* @param offCommand - 关命令
*/
public void setCommand(int no, Command onCommand, Command offCommand) {
this.onCommand[no] = onCommand;
this.offCommand[no] = offCommand;
}
/**
* 按下开按钮
* @param no 根据编号去调用哪个设备的执行方法
*/
public void onButton(int no) {
onCommand[no].execute();
}
/**
* 按下关按钮
* @param no 根据编号去调用哪个设备的执行方法
*/
public void offButton(int no) {
offCommand[no].execute();
}
}
⑥、测试
以上介绍只是写了一种设备,还可以直接添加其他设备,并不需要改动其他类。
结构图:
代码如下:
package com.lyd.demo.test;
import com.lyd.demo.command.light.LightOffCommand;
import com.lyd.demo.command.light.LightOnCommand;
import com.lyd.demo.command.light.LightReceive;
import com.lyd.demo.command.tv.TvOffCommand;
import com.lyd.demo.command.tv.TvOnCommand;
import com.lyd.demo.command.tv.TvReceive;
import com.lyd.demo.controller.RemoteController;
/**
* @Author: lyd
* @Description: 测试类
* @Date: 2022-09-03
*/
public class CommandTest {
public static void main(String[] args) {
// 创建灯光的接受者 - 具体方法类
LightReceive lightReceive = new LightReceive();
// 创建灯光的命令
LightOnCommand lightOnCommand = new LightOnCommand(lightReceive);
LightOffCommand lightOffCommand = new LightOffCommand(lightReceive);
// 创建遥控器
RemoteController remoteController = new RemoteController();
// 绑定命令到相应的按钮中
remoteController.setCommand(0, lightOnCommand, lightOffCommand); // 绑定
// 测试
System.out.println("按下开灯按钮》》》》》");
remoteController.onButton(0); // 0 代表是灯光按钮
System.out.println("按下关灯按钮》》》》》");
remoteController.offButton(0);
TvReceive tvReceive = new TvReceive();
TvOnCommand tvOnCommand = new TvOnCommand(tvReceive);
TvOffCommand tvOffCommand = new TvOffCommand(tvReceive);
remoteController.setCommand(1, tvOnCommand, tvOffCommand);
System.out.println("按下开启电视按钮》》》》》");
remoteController.onButton(1);
System.out.println("按下关闭电视按钮》》》》》");
remoteController.offButton(1);
}
}
运行结果:
👍创作不易,可能有些语言不是很通畅,如有错误请指正,感谢观看!记得一键三连哦!👍
今天的内容看起来并不简单,但实质上除了设计模式的思路,其实就是Java的面向对象知识,只要肯动手多敲,就容易理解。