2023-04-01  阅读(3)
原文作者:LoveLion 原文地址:https://blog.csdn.net/lovelion/article/details/8796736

5 撤销操作的实现

在命令模式中,我们可以通过调用一个命令对象的execute()方法来实现对请求的处理,如果需要撤销(Undo)请求,可通过在命令类中增加一个逆向操作来实现。

|   | 扩展 除了通过一个 逆向操作 来实现撤销(Undo)外,还可以通过 保存对象的历史状态 来实现撤销,后者可使用备忘录模式(MementoPattern)来实现。| | :-----: | :-----: | |    |  扩展 除了通过一个 逆向操作 来实现撤销(Undo)外,还可以通过 保存对象的历史状态 来实现撤销,后者可使用备忘录模式(MementoPattern)来实现。 |

下面通过一个简单的实例来学习如何使用命令模式实现撤销操作:

Sunny软件公司欲开发一个简易计算器,该计算器可以实现简单的数学运算,还可以对运算实施撤销操作。

Sunny软件公司开发人员使用命令模式设计了如图5所示结构图,其中计算器界面类CalculatorForm充当请求发送者,实现了数据求和功能的加法类Adder充当请求接收者,界面类可间接调用加法类中的add()方法实现加法运算,并且提供了可撤销加法运算的undo()方法。

202304012105314002.png

图5 简易计算器结构图

本实例完整代码如下所示:

    //加法类:请求接收者
    class Adder {
    	private int num=0; //定义初始值为0
    	
        //加法操作,每次将传入的值与num作加法运算,再将结果返回
    	public int add(int value) {
    		num += value;
    		return num;
    	}
    }
    
    //抽象命令类
    abstract class AbstractCommand {
    	public abstract int execute(int value); //声明命令执行方法execute()
    	public abstract int undo(); //声明撤销方法undo()
    }
    
    //具体命令类
    class ConcreteCommand extends AbstractCommand {
    	private Adder adder = new Adder();
    	private int value;
    		
    	//实现抽象命令类中声明的execute()方法,调用加法类的加法操作
    public int execute(int value) {
    		this.value=value;
    		return adder.add(value);
    	}
    	
        //实现抽象命令类中声明的undo()方法,通过加一个相反数来实现加法的逆向操作
    	public int undo() {
    		return adder.add(-value);
    	}
    }
    
    //计算器界面类:请求发送者
    class CalculatorForm {
    	private AbstractCommand command;
    	
    	public void setCommand(AbstractCommand command) {
    		this.command = command;
    	}
    	
        //调用命令对象的execute()方法执行运算
    	public void compute(int value) {
    		int i = command.execute(value);
    		System.out.println("执行运算,运算结果为:" + i);
    	}
    	
        //调用命令对象的undo()方法执行撤销
    	public void undo() {
    		int i = command.undo();
    		System.out.println("执行撤销,运算结果为:" + i);
    	}
    }

编写如下客户端测试代码:

    class Client {
    	public static void main(String args[]) {
    		CalculatorForm form = new CalculatorForm();
    		AbstractCommand command;
    		command = new ConcreteCommand();
    		form.setCommand(command); //向发送者注入命令对象
    		
    		form.compute(10);
    		form.compute(5);
    		form.compute(10);
    		form.undo();
    	}
    }

编译并运行程序,输出结果如下:

执行运算,运算结果为:10

执行运算,运算结果为:15

执行运算,运算结果为:25

执行撤销,运算结果为:15

需要注意的是在本实例中只能实现一步撤销操作,因为没有保存命令对象的历史状态,可以通过引入一个命令集合或其他方式来存储每一次操作时命令的状态,从而实现多次撤销操作。除了Undo操作外,还可以采用类似的方式实现恢复(Redo)操作,即恢复所撤销的操作(或称为 二次撤销 )。


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文