轉:C#中Undo/Redo的一個簡易實現

來源:互聯網
上載者:User

標籤:style   blog   http   color   使用   os   

一個比較常見的改進使用者體驗的方案是用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封裝的方式實現的也有一些限制,使用的時候需要注意:

  1. 每一步操作都需要封裝成command命令
  2. 每一步操作都是可逆的
  3. 當命令過多的時候需要考慮commandlist的記憶體佔用和命令查詢時的效能問題
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.