C#中通过Command模式实现Redo/Undo方案

作者:天方 时间:2021-07-12 14:58:59 

一个比较常见的改进用户体验的方案是用Redo/Undo来取代确认对话框,由于这个功能比较常用,本文简单的给了一个在C#中通过Command模式实现Redo/Undo方案的例子,以供后续查询。

class Program
{
   static void Main(string[] args)
   {
       var cmds = new CommandManager();
       while (true)
       {
           var key = Console.ReadKey(true);
           if (key.KeyChar >= '0' && key.KeyChar <= '9')
           {
               cmds.DoNewCommand(key.KeyChar.ToString(), () => Console.WriteLine("process " + key.KeyChar), () => Console.WriteLine("redo " + key.KeyChar));
           }
           else
           {
               if (key.Modifiers.HasFlag(ConsoleModifiers.Control) && (key.Key == ConsoleKey.Z))
                   cmds.UnDo();
               else if (key.Modifiers.HasFlag(ConsoleModifiers.Control) && (key.Key == ConsoleKey.Y))
                   cmds.ReDo();
           }
       }
   }
}

class CommandManager
{
   #region Command定义
   public class Command
   {
       string name;
       Action action;
       Action unDoAction;

internal Command(string name, Action action, Action unDoAction)
       {
           this.name = name;
           this.action = action;
           this.unDoAction = unDoAction;
       }

internal void Do() { action(); }
       internal void UnDo() { unDoAction(); }

public override string ToString()
       {
           return name.ToString();
       }
   }
   #endregion

public Stack<Command> ReDoActionStack { get; private set; }
   public Stack<Command> UnDoActionStack { get; private set; }

public CommandManager()
   {
       ReDoActionStack = new Stack<Command>();
       UnDoActionStack = new Stack<Command>();
   }

public void DoNewCommand(string name, Action action, Action unDoAction)
   {
       var cmd = new Command(name, action, unDoAction);
       UnDoActionStack.Push(cmd);
       ReDoActionStack.Clear();
       cmd.Do();
   }

public void UnDo()
   {
       if (!CanUnDo)
           return;

var cmd = UnDoActionStack.Pop();
       ReDoActionStack.Push(cmd);
       cmd.UnDo();
   }

public void ReDo()
   {
       if (!CanReDo)
           return;

var cmd = ReDoActionStack.Pop();
       UnDoActionStack.Push(cmd);
       cmd.Do();
   }

public bool CanUnDo { get { return UnDoActionStack.Count != 0; } }
   public bool CanReDo { get { return ReDoActionStack.Count != 0; } }
   //public IEnumerable<Command> Actions { get { return ReDoActionStack.Reverse().Concat(UnDoActionStack); } }
}

原理很简单,通过Command模式把每一步操作封装成一个可undo的命令(包含do和redo两个操作)。并将每一步操作执行后用栈保存起来,undo的时候就以此将Command依次出栈,并执行undo操作。(从某种意义上来说,redo就是undo操作的undo)

上面的代码已经实现了基本的Undo/Redo功能,但实际使用的时候还是有一些细节需要考虑的:如undo或redo时失败(抛异常)的处理等。由于这些细节方面的处理方式不尽相同,本文只是实现一个基本框架,以备后续使用时参考,并不想把它弄的过于复杂。

这种方式比较简单,几乎每种语言都可以轻易的写出这种方式下的实现。但通过这种Command封装的方式实现的也有一些限制,使用的时候需要注意:

  • 每一步操作都需要封装成command命令

  • 每一步操作都是可逆的

  • 当命令过多的时候需要考虑commandlist的内存占用和命令查询时的性能问题

来源:https://www.cnblogs.com/TianFang/archive/2010/10/05/1844385.html

标签:C#,Command,模式,Redo,Undo
0
投稿

猜你喜欢

  • maven中profile的使用

    2022-03-31 10:43:53
  • springboot项目以jar包运行的操作方法

    2023-11-10 07:47:43
  • Java8 使用工厂方法supplyAsync创建CompletableFuture实例

    2023-02-14 03:57:22
  • 聊聊Java的switch为什么不支持long

    2023-08-24 17:35:14
  • Android Studio和Gradle使用不同位置JDK的问题解决

    2023-06-27 17:35:04
  • 聊聊@RequestBody和Json之间的关系

    2023-11-27 03:31:45
  • Java实现二叉搜索树的插入、删除功能

    2023-07-15 20:54:53
  • Windows下Java环境配置的超详细教程

    2021-12-18 19:19:16
  • TextView实现跑马灯效果 就这么简单!

    2023-06-25 18:42:24
  • 浅析Java编程中枚举类型的定义与使用

    2021-07-04 23:46:16
  • PyQt5内嵌浏览器注入JavaScript脚本实现自动化操作的代码实例

    2023-11-26 15:05:59
  • Java实现简易扑克牌游戏的完整实例

    2022-10-08 14:09:18
  • Spring Boot利用JSR303实现参数验证的方法实例

    2022-07-28 20:36:26
  • kill命令在Java应用中使用的注意事项小结

    2023-11-11 13:01:55
  • 一篇带你解析入门LongAdder源码

    2023-11-28 20:17:52
  • Java中switch的三种用法方式小结

    2023-11-24 03:40:21
  • Java创建子线程的两种方法

    2023-11-24 07:00:05
  • SparkSQL开窗函数分析使用示例

    2022-04-16 02:26:32
  • Java编程实现帕斯卡三角形代码示例

    2023-11-02 08:08:24
  • javax.persistence中@Column定义字段类型方式

    2021-12-03 21:21:44
  • asp之家 软件编程 m.aspxhome.com