Как глубоко клонировать объекты, содержащие использование свойства IList AutoMapper

Я пытаюсь глубоко клонировать следующее использование класса AutoMapper:

public class MainData
{
    public MainData()
    {
        Details = new List<Detail>();
    }

    public int Id { get; private set; }
    public DateTime LastUpdate { get; private set; }
    public IList<Detail> Details { get; private set; }
    public int Prop1 { get; set; }
    public int Prop2 { get; set; }

    public void AddDetail(Detail detail)
    {
        Details.Add(detail);
    }

    public void RemoveDetail(Detail detail)
    {
        Details.Remove(detail);
    }

    public MainData Clone()
    {
        Mapper.Reset();
        Mapper.CreateMap<MainData, MainData>().ForMember(d => d.Id, o => o.Ignore());
        // Mapper.CreateMap<Detail, Detail>().ForMember(d => d.Id, o => o.Ignore()); // REMOVED
        var newMainData = new MainData();
        Mapper.Map(this, newMainData);
        newMainData.Details = this.Details.Select(item => item.Clone()).ToList(); // ADDED
        return newMainData;
    }
}

public class Detail
{
    public int Id { get; private set; }
    public string Name { get; set; }
    public double Area { get; set; }
    public double Height { get; set; }

    public Detail Clone() // ADDED
    {
        Mapper.CreateMap<Detail, Detail>().ForMember(d => d.Id, o => o.Ignore());
        var newDetail = new Detail();
        Mapper.Map(this, newDetail);
        return newDetail;
    }
}

Clone метод хорошо работает для свойств MainData, но, кажется, только делает мелкую копию списка Деталей. Я попытался добавить .ForMember(d => d.Details, o => o.UseDestinationValue()) но это не копирует список Деталей вообще. Как я могу получить список Деталей, глубоко клонированный также т.е., таким образом, я заканчиваю с двумя полностью независимыми объектами включая все элементы списка?

ОБНОВЛЕНИЕ: Я должен исключить свойство Id, поскольку я использую эти объекты с NHibernate, таким образом, не уверенным, если сериализуемое решение сделает это.

UPDATE2: Измененный вышеупомянутый код для клонирования IList также. Это, кажется, хорошо работает, поскольку я могу исключить свойства, которые заставляют NHibernate думать, что он был уже сохранен.

13
задан JasonMArcher 8 July 2015 в 04:30
поделиться

2 ответа

вот одно решение с ValueInjecter

        var clone = new MainData();

        clone.InjectFrom(mainData);//mainData is your source

        mainData.Details.AsParallel.ForAll(detail => 
        {
            var dc = new Detail();
            dc.InjectFrom(detail);
            clone.AddDetail(dc);
        });

, свойства, у которых есть частные сеттеры, не будут устанавливаться (выглядит разумным)
удачи;)

РЕДАКТИРОВАТЬ: Я сделал лучшее решение, посмотрите здесь

9
ответ дан 1 December 2019 в 23:30
поделиться

AutoMapper на самом деле не является API клонирования. Вместо этого я бы использовал этот трюк клонирования:

public static object CloneObject(object obj)
{
    using (MemoryStream memStream = new MemoryStream())
    {
        BinaryFormatter binaryFormatter = new BinaryFormatter(null, 
             new StreamingContext(StreamingContextStates.Clone));
        binaryFormatter.Serialize(memStream, obj);
        memStream.Seek(0, SeekOrigin.Begin);
        return binaryFormatter.Deserialize(memStream);
    }
}

Он работает не во всех ситуациях, но довольно удобен.

9
ответ дан 1 December 2019 в 23:30
поделиться
Другие вопросы по тегам:

Похожие вопросы: