【问题标题】:Implementing undo/redo using MemberwiseClone使用 MemberwiseClone 实现撤消/重做
【发布时间】:2011-03-10 08:41:48
【问题描述】:

我正在尝试在我正在处理的 C# 应用程序中实现撤消/重做堆栈,方法是在调用撤消时将对象恢复到以前的状态。我有一个“Action”类,它看起来基本上是这样的:

class Action
{
    object old_state;
    object new_state;

    public Action(object old)
    {
        old_state = old;
    }

    public void finish(object new_obj)
    {
        new_state = new_obj;
    }

    public void reverse()
    {
        new_state = old_state;
    }
}

当开始一个可以重做的动作时,会创建一个新动作。当我们到达新状态时,finish() 被调用。当用户想要重做某事时,它会调用 reverse() 并将 Object 恢复到其原始状态。

显然这不起作用,因为两个对象都是通过引用传递的,并且对象只是以新状态结束。

我真正想做的是能够说:

public Action(object old)
{
    old_state = old.MemberwiseClone();
}

不幸的是,这不起作用,我收到如下错误:

无法通过“foo.object”类型的限定符访问受保护成员“object.MemberwiseClone()”

我想创建原始状态的浅表副本(按值复制所有值字段,按引用复制所有引用字段),但我不知道如何使用通用对象来做到这一点,相反在我可能希望恢复状态的每个类中实现IClonable

谁能提供任何见解?

【问题讨论】:

标签: c# clone


【解决方案1】:

您可以使用此版本的克隆对象(注意:对象必须可序列化才能使用此功能):

  /// <summary>
  /// Clones Any Object.
  /// </summary>
  /// <param name="objectToClone">The object to clone.</param>
  /// <return>The Clone</returns>
  public static T Clone<T>(T objectToClone)
  {
     T cloned_obj = default(T);
     if ((!Object.ReferenceEquals(objectToClone, null)) && (typeof(T).IsSerializable))
     {
        System.Runtime.Serialization.Formatters.Binary.BinaryFormatter bin_formatter = null;
        Byte[] obj_bytes = null;

        using (MemoryStream memory_stream = new MemoryStream(1000))
        {
           bin_formatter = new System.Runtime.Serialization.Formatters.Binary.BinaryFormatter();
           try
           {
              bin_formatter.Serialize(memory_stream, objectToClone);
           }
           catch (Exception) { }
           obj_bytes = memory_stream.ToArray();
        }

        using (MemoryStream memory_stream = new MemoryStream(obj_bytes))
        {
           try
           {
              cloned_obj = (T)bin_formatter.Deserialize(memory_stream);
           }
           catch (Exception) { }
        }
     }
     return cloned_obj;
  }

【讨论】:

  • 非常感谢——这很有帮助
【解决方案2】:

要对该问题添加更多见解...我注意到您不能使用实例引用来调用该方法(因为 MemberwiseClone() 是使用受保护的范围标识符定义的)。相反,您必须使用 this 关键字。

对于上面给出的示例,您必须这样做:

public Action(object old)
{
    old_state = this.MemberwiseClone();
}

【讨论】:

    猜你喜欢
    • 2021-06-04
    • 2012-06-02
    • 2017-03-16
    • 2016-02-18
    • 1970-01-01
    • 2011-11-14
    • 2012-07-02
    • 2014-04-20
    • 2011-12-03
    相关资源
    最近更新 更多