命令模式

在这里插入图片描述
模拟一个多功能遥控器,一件开关灯和空调。
如果不使用命令模式,遥控器与电器耦合在一起,如果再增加一个电器,就需要修改遥控器类,违反开闭原则。
如果不使用命令模式也不能实现undo和redo功能。

package command;

public interface Appliance {
	
	void on();
	
	void off();
}

package command;

public class Light implements Appliance{
	
	@Override
	public void on() {
		System.out.println("开灯");
		
	}

	@Override
	public void off() {
		System.out.println("关灯");
	}
	
}

package command;

public class AirConditioner implements Appliance{

	@Override
	public void on() {
		System.out.println("打开空调");
	}

	@Override
	public void off() {
		System.out.println("关闭空调");
	}

}

package command;

public abstract class Command {
	
	Appliance appliance;

	public Command(Appliance appliance) {
		super();
		this.appliance = appliance;
	}
	
	//执行
	public abstract void execute();
	
	//撤销
	public abstract void undo();
	
}

package command;

public class LightOnCommand extends Command{

	public LightOnCommand(Light light) {
		super(light);
	}

	@Override
	public void execute() {
		appliance.on();
	}

	@Override
	public void undo() {
		appliance.off();
	}

}

package command;

public class LightOffCommand extends Command{

	public LightOffCommand(Light light) {
		super(light);
	}

	@Override
	public void execute() {
		appliance.off();
	}

	@Override
	public void undo() {
		appliance.on();
	}

}

package command;

public class AirConditionerOnCommand extends Command{

	public AirConditionerOnCommand(Appliance appliance) {
		super(appliance);
	}

	@Override
	public void execute() {
		appliance.on();
	}

	@Override
	public void undo() {
		appliance.off();
	}

}

package command;

public class AirConditionerOffCommand extends Command{

	public AirConditionerOffCommand(Appliance appliance) {
		super(appliance);
	}

	@Override
	public void execute() {
		appliance.off();
	}

	@Override
	public void undo() {
		appliance.on();
	}

}

package command;

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;

public class RemoteInvoker {
	
	//开命令
	private List<Command> onCommands=new ArrayList<>();
	
	//关命令
	private List<Command> offCommands=new ArrayList<>();
	
	//撤回命令
	private Stack<List<Command>> undoCommands=new Stack<>();
	
	//重新执行命令
	private Stack<List<Command>> redoCommands=new Stack<>();
	
	//添加开命令
	public void addOnCommand(Command onCommand) {
		onCommands.add(onCommand);
	}
	
	//添加关命令
	public void addOffCommand(Command offCommand) {
		offCommands.add(offCommand);
	}
	
	//打开开关,执行开命令
	public void pressOnButton() {
		onCommands.stream().forEach(Command::execute);
		undoCommands.push(onCommands);
		redoCommands.clear();
	}
	
	//关闭开关,执行关命令
	public void pressOffButton() {
		offCommands.stream().forEach(Command::execute);
		undoCommands.push(offCommands);
		redoCommands.clear();
	}
	
	//撤销
	public void undo() {
		List<Command> commands=undoCommands.pop();
		commands.stream().forEach(Command::undo);
		redoCommands.push(commands);
	}
	
	//重新执行
	public void redo() {
		List<Command> commands=redoCommands.pop();
		commands.stream().forEach(Command::execute);
		undoCommands.push(commands);
	}
}

package command;

public class CommandTest {
	public static void main(String[] args) {
		
		//demo的构建思路:
		//1.构建单命令demo
		//2.添加多命令
		//3.添加多命令单回退
		//4.添加多命令多回退
		//5.添加重新执行
		
		Light light=new Light();
		AirConditioner airConditioner=new AirConditioner();
		
		Command lightOnCommand=new LightOnCommand(light);
		Command lightOffCommand=new LightOffCommand(light);
		Command airConditionerOnCommand=new AirConditionerOnCommand(airConditioner);
		Command airConditionerOffCommand=new AirConditionerOffCommand(airConditioner);
		
		RemoteInvoker remoteInvoker=new RemoteInvoker();
		remoteInvoker.addOnCommand(lightOnCommand);
		remoteInvoker.addOffCommand(lightOffCommand);
		remoteInvoker.addOnCommand(airConditionerOnCommand);
		remoteInvoker.addOffCommand(airConditionerOffCommand);
		
		remoteInvoker.pressOnButton();
		remoteInvoker.pressOffButton();
		remoteInvoker.pressOnButton();
		remoteInvoker.pressOffButton();
		
		remoteInvoker.undo();
		remoteInvoker.undo();
		remoteInvoker.undo();
		remoteInvoker.undo();
		remoteInvoker.redo();
		remoteInvoker.pressOffButton();
	}
}

打印结果:
开灯
打开空调
关灯
关闭空调
开灯
打开空调
关灯
关闭空调
开灯
打开空调
关灯
关闭空调
开灯
打开空调
关灯
关闭空调
开灯
打开空调
关灯
关闭空调

小结:
1.命令模式的好处是将调用者(Invoker)与实际实现功能的对象进行解耦,同时能够实现撤销和重做的功能。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值