有限次数的Undo&Redo的C#实现
为了实现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操作的状态。
- 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()方法。
- /// <summary>
- /// 实现Undo操作
- /// </summary>
- /// <param name="times">撤销的次数</param>
- public static void Undo()
- {
- if (undoStack.Count != 0)
- {
- MyCommand myCommand = undoStack.Pop();
- myCommand.undo();
- redoStack.Push(myCommand);
- }
- }
在Redo方法中:
检查Redo栈是否为空,不为空,则弹出一个MyCommand对象,把这个对象压入Undo栈中,执行这个对象的execute()方法。
- /// <summary>
- /// 实现Redo操作
- /// </summary>
- /// <param name="times">撤销的次数</param>
- public static void Redo()
- {
- if (redoStack.Count != 0)
- {
- MyCommand myCommand = redoStack.Pop();
- myCommand.execute();
- undoStack.Push(myCommand);
- }
- }
在向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的C#实现的更多相关文章
- MySQL,MariaDB:Undo | Redo [转]
本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...
- iOS: 为画板App增加 Undo/Redo(撤销/重做)操作
这个随笔的内容以上一个随笔为基础,(在iOS中实现一个简单的画板),上一个随笔实现了一个简单的画板: 今天我们要为这个画板增加Undo/Redo操作,当画错了一笔,可以撤销它,或者撤销之后后悔了, ...
- [转]MySQL日志——Undo | Redo
本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...
- 从Undo,Redo谈命令模式
一般的应用软件中,通常会提供Redo和Undo的操作,比如Paint.NET中的动作面板,Word中的撤销重做,一般我们按Ctrl-Z即可回退到上次操作. 要实现上面的这一功能,最直观的想法就是,我们 ...
- 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 ...
- 【转载】MySQL 日志 undo | redo
本文是介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版 ...
- MySQL InnoDB存储引擎undo redo解析
本文介绍MySQL数据库InnoDB存储引擎重做日志漫游 00 – Undo Log Undo Log 为了实现事务原子,在MySQL数据库InnoDB存储引擎,还使用Undo Log(简称:MVCC ...
- MySQL日志Undo&Redo
00 – Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版本并发控制(简称:MVCC). - 事务的原子性(Atomi ...
- MySql Undo Redo
Undo LogUndo Log 是为了实现事务的原子性,在MySQL数据库InnoDB存储引擎中,还用Undo Log来实现多版本并发控制(简称:MVCC). - 事务的原子性(Atomicity) ...
随机推荐
- python 实时监控剪切板,并替换其中的部分内容,重新写入剪切板
#实时监控剪贴板内容的变化,并替换其中的回车,换行,逗号,再写入剪切板,以供使用. import pyperclip import time last_string = pyperclip.paste ...
- vue-qiankun公司微前端项稳定目落地后的总结(附github仓库demo,将会持续更新)
️本文为博客园社区首发文章,未获授权禁止转载 大家好,我是aehyok,一个住在深圳城市的佛系码农♀️,如果你喜欢我的文章,可以通过点赞帮我聚集灵力️. 个人github仓库地址: https:gi ...
- Requests方法 --- post 请求body的四种类型
常见的 post 提交数据类型有四种: 1.第一种:application/json:这是最常见的 json 格式,也是非常友好的深受小伙伴喜欢的一种,如下{"input1":&q ...
- 随机数种子(random seed)
在科学技术和机器学习等其他算法相关任务中,我们经常需要用到随机数,为了把握随机数的生成特性,从随机数的随机无序中获得确定和秩序.我们可以利用随机数种子(random seed)来实现这一目标,随机数种 ...
- Optional 的使用会导致性能下降吗?
几天前,我在论坛上发了一篇关于Optional 的文章.其中一条评论是一个非常好的问题: Optional 的使用会导致性能下降吗? 答案是: 是的,它会的.但是你应该担心吗? 使用Optional的 ...
- React refs 的理解
一.是什么 Refs 在计算机中称为弹性文件系统(英语:Resilient File System,简称ReFS) React 中的 Refs提供了一种方式,允许我们访问 DOM节点或在 render ...
- P5147-数学-随机数生成器
P5147-数学-随机数生成器 (洛谷第一篇题解说这是高一数学题,新高二感觉到被吊打) 我们设work(x)的期望值为\(f_x\) 注意\(f_1\)是边界.不过对下列式子没有影响.原因参照必修的数 ...
- 【洛谷P1140 相似基因】动态规划
分析 f[i][j] 表示 1数组的第i位和2数组的第j位匹配的最大值 f[1][1]=-2 f[2][1]=-2+5=3 f[3][1]=-2+5+5=8 三个决策: 1.由f[i-1][j-1]直 ...
- 货币兑换问题(贪心法)——Python实现
# 贪心算法求解货币兑换问题 # 货币系统有 n 种硬币,面值为 v1,v2,v3...vn,其中 v1=1,使用总值money与之兑换,求如何使硬币的数目最少,即 x1,x2,x3...xn 之 ...
- frameset框架在.net网站中的小实现。
一般我们生成网页,为减少代码的开发量,通常将不变的网页部分进行重用.通用为三种方法: 1.frameset框架 2.用户自定义控件 3.母版页(消耗资源大,不追叙) 通常1,2两种方法常用. 1.fr ...