Клонирование ссылочного типа C # в производный ссылочный тип - PullRequest
4 голосов
/ 26 августа 2009

Исходя из фона C ++, я нахожу клонирование объектов в C # немного сложным для привыкания. Чтобы избавиться от некоторых недоразумений, я ищу элегантный способ клонирования объекта базового типа в производный тип.

Для иллюстрации:

public class Base
{
    public string Member1;
    public int Member2;
    public float Member3;
    public bool Member4;
}

public class Derived : Base
{
    public List<Base> Children;
}

Base base = new Base();

И с этим я хочу создать экземпляр «Derived», одновременно делая для каждого элемента копию базового объекта - желательно без назначения их вручную.

Примечание. Может быть, это больше подходит для типа значения?

Ответы [ 2 ]

4 голосов
/ 26 августа 2009

Поскольку вы не можете изменить тип объекта, у вас есть несколько вариантов:

  • инкапсулируют Base
  • используйте конструктор, который копирует значения из Base
  • скопировать свойства из Base через отражение или подобное

Для последнего MiscUtil имеет полезный инструмент:

Base b= ...
Derived item = PropertyCopy<Derived>.CopyFrom(b);

Для инкапсуляции:

public class Derived
{
    readonly Base b;
    public Derived(Base b) {this.b=b;}
    public List<Base> Children;
    public string Member1 {get {return b.Member1;} set {...} }
    public int Member2 {etc}
    public float Member3 {etc}
    public bool Member4 {etc}
}

Или как ручная копия:

public class Derived : Base
{
    public Derived(Base b) {
        this.Member1 = b.Member1;
        // etc
    }
    // additional members...
}

или (комментарии) получить базу для копирования себя:

public class Derived : Base
{
    public Derived(Base b) : base(b) { }
    // additional members...
}
public class Base
{
    // members not shown...
    public Base() {}
    protected Base(Base b) {
       this.Member1 = b.Member1;
        // etc
    }
    // additional members...
}

(где конструктор Base инициализирует поля с Base)

2 голосов
/ 07 ноября 2011
/// Clone all fields from an instance of base class TSrc into derived class TDst
public static TDst Clone<TSrc, TDst>(TSrc source, TDst target)
    where TDst : TSrc
{
    var bf = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.FlattenHierarchy;
    foreach (FieldInfo fis in source.GetType().GetFields(bf))
        fis.SetValue(target, fis.GetValue(source));
    return target;
}

/// Create a new instance of a derived class, cloning all fields from type TSrc
public static TDst Clone<TSrc, TDst>(TSrc source)
    where TDst : TSrc, new()
{
    return Clone(source, new TDst());
}
...