为了实现Undo和Redo,必须要在程序中保存起程序的运行状态,从而能够在Undo时跳转到前一个状态和在Redo时跳转到下一个状态。为了实现状态的维护,我采用了两个栈来分别保存Undo操作的状态和Redo操作的状态。

public static Stack<MyCommand> undoStack = new Stack<MyCommand>();

public static Stack<MyCommand> redoStack = new Stack<MyCommand>();

首先要识别哪些操作可以支持Undo和Redo操作。在我的小程序中,支持的操作主要有几个:textbox的textchanged,textbox和button的焦点,radiobutton、checkbox、combox、listbox选项的改变。

对于上述操作的实现,必须要实现一个MyCommand接口。

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

namespace UndoRedo

{

public interface MyCommand

{

void execute(); //完成动作

void undo(); //撤销动作

}

}

每个操作都要继承自这个MyCommand接口,在操作类中包含有实现Undo和Redo操作所需要的属性,并且实现了接口中的execute()和undo()

创建了一个UndoRedo类,类中包含上面提到的两个栈,一个Undo栈,一个Redo栈。这个类实现了Undo方法和Redo方法,并且还有多个向Undo栈进行压栈的方法。

在Undo方法中:

检查Undo栈是否为空,不为空,则弹出一个MyCommand对象,把这个对象压入Redo栈中,执行这个对象的undo()方法。

00001.

/// <summary>

00002.

00003.

/// 实现Undo操作

00004.

00005.

/// </summary>

00006.

00007.

/// <param name="times">撤销的次数</param>

00008.

00009.

public static void Undo()

00010.

00011.

{

00012.

00013.

if (undoStack.Count != 0)

00014.

00015.

{

00016.

00017.

MyCommand myCommand = undoStack.Pop();

00018.

00019.

myCommand.undo();

00020.

00021.

redoStack.Push(myCommand);

00022.

00023.

}

00024.

00025.

}

00026.

在Redo方法中:

检查Redo栈是否为空,不为空,则弹出一个MyCommand对象,把这个对象压入Undo栈中,执行这个对象的execute()方法。

00001.

/// <summary>

00002.

00003.

/// 实现Redo操作

00004.

00005.

/// </summary>

00006.

00007.

/// <param name="times">撤销的次数</param>

00008.

00009.

public static void Redo()

00010.

00011.

{

00012.

00013.

if (redoStack.Count != 0)

00014.

00015.

{

00016.

00017.

MyCommand myCommand = redoStack.Pop();

00018.

00019.

myCommand.execute();

00020.

00021.

undoStack.Push(myCommand);

00022.

00023.

}

00024.

00025.

}

00026.

在向Undo栈进行压栈的方法中:

将MyCommand对象压入Undo栈中,并且将Redo栈清空。在这个方法里需要注意一点的是,我是实现有限次数的Undo和Redo,所以将栈的大小必须控制起来。如果栈中的元素个数小于指定次数,则进行压栈操作;如果栈中元素等于指定次数,则将栈中元素进行了一个处理。我是这样处理的:将栈内的元素用一个list保存起来,并且将除了栈底元素外的其他元素都重新压回栈内,从而实现了栈的元素个数的有限。下面这段代码以textbox的text改变事件作为例子,其他操作类似。

public static void dealWithUndoStack(MyCommand command)

{

List<MyCommand> commandList = new List<MyCommand>();

for (int i = 0; i < undoTimes; i++)

{

MyCommand cmd = undoStack.Pop();

commandList.Add(cmd);

}

for (int j = undoTimes - 2; j >= 0; j--)

{

undoStack.Push(commandList[j]);

}

}

/// <summary>

/// 字符串的修改

/// </summary>

/// <param name="nstr">新字符串</param>

public static void inStackForText(TextBox tb,string nstr,string ostr)

{

MyCommand command = new TextChangeCommand(tb,nstr,ostr);

if (undoStack.Count < undoTimes)

undoStack.Push(command);

else if (undoStack.Count == undoTimes)

dealWithUndoStack(command);

redoStack.Clear();

}

在完成了上面的几个步骤后,只需要在执行程序的不同操作的时候将该操作对应的Command类通过与inStackForText类似的方法,将类的对象压入Undo栈即可。当需要执行Undo操作的时候,调用UndoRedo类中的Undo方法;当需要执行Redo操作的时候,调用UndoRedo类中的Redo方法。

接着就是对于不同的操作,为其生成一个继承MyCommand接口的类即可。下面举个例子,依然是上面提到的textbox的text改变事件。

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

using System.Windows.Forms;

namespace UndoRedo

{

class TextChangeCommand : MyCommand

{

private string newStr;

private string oldStr;

private TextBox mTextbox;

public TextChangeCommand(TextBox tb,string ntext,string otext)

{

this.newStr = ntext;

this.mTextbox = tb;

this.oldStr = otext;

}

public void execute()

{

mTextbox.Text = this.newStr;

}

public void undo()

{

mTextbox.Text = this.oldStr;

}

}

}

这样的类的实现很简单,只需要将特定某类操作的操作对象和前后状态保存起来,并且实现接口中的方法即可。

总结一下:这样实现的好处就是不必把所需要用到Undo&Redo操作的控件的状态全保存起来,仅保存那一类操作所需的属性即可,让程序的可扩展性更好。当程序需要实现的功能发生改变的时候,只需要再实现一个继承自MyCommand接口的操作类,在UndoRedo类中为其生成一个压栈操作的方法即可。

为了实现Undo和Redo,必须要在程序中保存起程序的运行状态,从而能够在Undo时跳转到前一个状态和在Redo时跳转到下一个状态。为了实现状态的维护,我采用了两个栈来分别保存Undo操作的状态和Redo操作的状态。

  1.  
    public static Stack<MyCommand> undoStack = new Stack<MyCommand>();
  2.  
    public static Stack<MyCommand> redoStack = new Stack<MyCommand>();

首先要识别哪些操作可以支持Undo和Redo操作。在我的小程序中,支持的操作主要有几个:textbox的textchanged,textbox和button的焦点,radiobutton、checkbox、combox、listbox选项的改变。

对于上述操作的实现,必须要实现一个MyCommand接口。

  1.  
    using System;
  2.  
    using System.Collections.Generic;
  3.  
    using System.Linq;
  4.  
    using System.Text;
  5.  
     
  6.  
    namespace UndoRedo
  7.  
    {
  8.  
    public interface MyCommand
  9.  
    {
  10.  
    void execute(); //完成动作
  11.  
    void undo(); //撤销动作
  12.  
    }
  13.  
    }

每个操作都要继承自这个MyCommand接口,在操作类中包含有实现Undo和Redo操作所需要的属性,并且实现了接口中的execute()和undo()

创建了一个UndoRedo类,类中包含上面提到的两个栈,一个Undo栈,一个Redo栈。这个类实现了Undo方法和Redo方法,并且还有多个向Undo栈进行压栈的方法。

在Undo方法中:

检查Undo栈是否为空,不为空,则弹出一个MyCommand对象,把这个对象压入Redo栈中,执行这个对象的undo()方法。

  1.  
    /// <summary>
  2.  
    /// 实现Undo操作
  3.  
    /// </summary>
  4.  
    /// <param name="times">撤销的次数</param>
  5.  
    public static void Undo()
  6.  
    {
  7.  
    if (undoStack.Count != 0)
  8.  
    {
  9.  
    MyCommand myCommand = undoStack.Pop();
  10.  
    myCommand.undo();
  11.  
    redoStack.Push(myCommand);
  12.  
    }
  13.  
    }

在Redo方法中:

检查Redo栈是否为空,不为空,则弹出一个MyCommand对象,把这个对象压入Undo栈中,执行这个对象的execute()方法。

  1.  
    /// <summary>
  2.  
    /// 实现Redo操作
  3.  
    /// </summary>
  4.  
    /// <param name="times">撤销的次数</param>
  5.  
    public static void Redo()
  6.  
    {
  7.  
    if (redoStack.Count != 0)
  8.  
    {
  9.  
    MyCommand myCommand = redoStack.Pop();
  10.  
    myCommand.execute();
  11.  
    undoStack.Push(myCommand);
  12.  
    }
  13.  
    }

在向Undo栈进行压栈的方法中:

将MyCommand对象压入Undo栈中,并且将Redo栈清空。在这个方法里需要注意一点的是,我是实现有限次数的Undo和Redo,所以将栈的大小必须控制起来。如果栈中的元素个数小于指定次数,则进行压栈操作;如果栈中元素等于指定次数,则将栈中元素进行了一个处理。我是这样处理的:将栈内的元素用一个list保存起来,并且将除了栈底元素外的其他元素都重新压回栈内,从而实现了栈的元素个数的有限。下面这段代码以textbox的text改变事件作为例子,其他操作类似。

  1.  
    public static void dealWithUndoStack(MyCommand command)
  2.  
    {
  3.  
    List<MyCommand> commandList = new List<MyCommand>();
  4.  
    for (int i = 0; i < undoTimes; i++)
  5.  
    {
  6.  
    MyCommand cmd = undoStack.Pop();
  7.  
    commandList.Add(cmd);
  8.  
    }
  9.  
    for (int j = undoTimes - 2; j >= 0; j--)
  10.  
    {
  11.  
    undoStack.Push(commandList[j]);
  12.  
    }
  13.  
    }
  14.  
     
  15.  
    /// <summary>
  16.  
    /// 字符串的修改
  17.  
    /// </summary>
  18.  
    /// <param name="nstr">新字符串</param>
  19.  
    public static void inStackForText(TextBox tb,string nstr,string ostr)
  20.  
    {
  21.  
    MyCommand command = new TextChangeCommand(tb,nstr,ostr);
  22.  
    if (undoStack.Count < undoTimes)
  23.  
    undoStack.Push(command);
  24.  
    else if (undoStack.Count == undoTimes)
  25.  
    dealWithUndoStack(command);
  26.  
    redoStack.Clear();
  27.  
    }

在完成了上面的几个步骤后,只需要在执行程序的不同操作的时候将该操作对应的Command类通过与inStackForText类似的方法,将类的对象压入Undo栈即可。当需要执行Undo操作的时候,调用UndoRedo类中的Undo方法;当需要执行Redo操作的时候,调用UndoRedo类中的Redo方法。

接着就是对于不同的操作,为其生成一个继承MyCommand接口的类即可。下面举个例子,依然是上面提到的textbox的text改变事件。

  1.  
    using System;
  2.  
    using System.Collections.Generic;
  3.  
    using System.Linq;
  4.  
    using System.Text;
  5.  
    using System.Windows.Forms;
  6.  
     
  7.  
    namespace UndoRedo
  8.  
    {
  9.  
    class TextChangeCommand : MyCommand
  10.  
    {
  11.  
    private string newStr;
  12.  
    private string oldStr;
  13.  
    private TextBox mTextbox;
  14.  
     
  15.  
    public TextChangeCommand(TextBox tb,string ntext,string otext)
  16.  
    {
  17.  
    this.newStr = ntext;
  18.  
    this.mTextbox = tb;
  19.  
    this.oldStr = otext;
  20.  
    }
  21.  
     
  22.  
    public void execute()
  23.  
    {
  24.  
    mTextbox.Text = this.newStr;
  25.  
    }
  26.  
     
  27.  
    public void undo()
  28.  
    {
  29.  
    mTextbox.Text = this.oldStr;
  30.  
    }
  31.  
    }
  32.  
    }

这样的类的实现很简单,只需要将特定某类操作的操作对象和前后状态保存起来,并且实现接口中的方法即可。

总结一下:这样实现的好处就是不必把所需要用到Undo&Redo操作的控件的状态全保存起来,仅保存那一类操作所需的属性即可,让程序的可扩展性更好。当程序需要实现的功能发生改变的时候,只需要再实现一个继承自MyCommand接口的操作类,在UndoRedo类中为其生成一个压栈操作的方法即可。

有限次数的Undo&Redo的C#实现的更多相关文章

  1. MySQL,MariaDB:Undo | Redo [转]

    本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...

  2. iOS: 为画板App增加 Undo/Redo(撤销/重做)操作

    这个随笔的内容以上一个随笔为基础,(在iOS中实现一个简单的画板),上一个随笔实现了一个简单的画板:   今天我们要为这个画板增加Undo/Redo操作,当画错了一笔,可以撤销它,或者撤销之后后悔了, ...

  3. [转]MySQL日志——Undo | Redo

    本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...

  4. 从Undo,Redo谈命令模式

    一般的应用软件中,通常会提供Redo和Undo的操作,比如Paint.NET中的动作面板,Word中的撤销重做,一般我们按Ctrl-Z即可回退到上次操作. 要实现上面的这一功能,最直观的想法就是,我们 ...

  5. Undo/Redo for Qt Tree Model

    Undo/Redo for Qt Tree Model eryar@163.com Abstract. Qt contains a set of item view classes that use ...

  6. 【转载】MySQL 日志 undo | redo

    本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...

  7. MySQL InnoDB存储引擎undo redo解析

    本文介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo Log Undo Log 为了实现事务原子,在MySQL数据库InnoDB存储引擎,还使用Undo Log(简称:MVCC ...

  8. MySQL日志Undo&Redo

    00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版本并发控制(简称:MVCC). - 事务的原子性(Atomi ...

  9. MySql Undo Redo

    Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版本并发控制(简称:MVCC). - 事务的原子性(Atomicity) ...

随机推荐

  1. 利用扫描仪形成PDF

    1.打开WPS,新建PDF,从扫描仪新建 2.合并PDF:按照顺序添加指定PDF,合并即可完成

  2. asp.net web.config数据库连接字符串加密与解密

    在WEB网站开发过程中,如果我们将数据库连接字符串封装到.DLL文件中,将会给数据库和程序的迁移带来麻烦,因为万一服务器地址或者数据库发生变更,那么我们就不得不修改源程序并重新将其编译.最好的解决方法 ...

  3. Modelsim波形显示字符

    偶然在 QQ 群里看到一个大佬发的 Modelsim 波形显示字符,闲着没事拿来玩玩,并将改良过程也整理一下. 一.字符点阵产生 软件采用 PCtoLCD2002,打开后不需要设置,直接打字然后点击[ ...

  4. ZooKeeper 分布式锁 Curator 源码 04:分布式信号量和互斥锁

    前言 分布式信号量,之前在 Redisson 中也介绍过,Redisson 的信号量是将计数维护在 Redis 中的,那现在来看一下 Curator 是如何基于 ZooKeeper 实现信号量的. 使 ...

  5. React组件三大属性之state

    React组件三大属性之state 组件被称为"状态机", 页面的显示是根据组件的state属性的数据来显示 理解1) state是组件对象最重要的属性, 值是对象(可以包含多个数 ...

  6. FreeRTOS-01-任务相关函数

    3 任务相关API函数 任务相关函数如下: 任务创建和删除API函数 任务创建和删除实验(动态方法) 任务创建和删除实验(静态方法) 任务挂起和恢复API函数 任务挂起和恢复实验 3.1 任务创建AP ...

  7. VSCode 如何远程连接其他主机的 WSL2

    VSCode 如何远程连接其他主机的 WSL2 VSCode 的 Remote Deployment 插件对 WSL2 直接提供了支持,能够很方便的连接本机的 WSL2 ,但是并没有提供一个连接远程 ...

  8. Python如何将py文件打包成exe

    安装pyinstaller 打开cmd窗口,输入pip install pyinstaller,命令行输出successfully表示成功. 生成exe文件 一.单个py文件 在py文件目录下,打开c ...

  9. Python自动化测试面试题-Linux篇

    目录 Python自动化测试面试题-经验篇 Python自动化测试面试题-用例设计篇 Python自动化测试面试题-Linux篇 Python自动化测试面试题-MySQL篇 Python自动化测试面试 ...

  10. 旧VC项目dpiAware支持

    起因 工作原因,需要维护一款VS2008 SP1开发的MFC项目, 发现WIN10高分辨率下显示模糊,不考虑升级VC版本情况下尝试解决 尝试 新版本VC中Manifest Tool>Input ...