14

我正在尝试使用 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 属性,但似乎只对 Details 列表进行了浅拷贝。我尝试添加.ForMember(d => d.Details, o => o.UseDestinationValue()),但这根本不会复制详细信息列表。如何将详细信息列表也进行深度克隆,即最终得到两个完全独立的对象,包括所有列表项?

更新:我需要排除 Id 属性,因为我将这些对象与 NHibernate 一起使用,所以不确定 Serializable 解决方案是否会这样做。

UPDATE2:修改了上面的代码来克隆IList。这似乎工作正常,因为我可以排除使 NHibernate 认为它已经被保存的属性。

4

2 回答 2

9

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);
    }
}

它并不适用于所有情况,但它非常方便。

于 2010-08-03T13:35:50.017 回答
9

这是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);
        });

不会设置具有私有设置器的属性,(看起来很合理)
祝你好运;)

编辑:我在这里做了一个更好的解决方案

于 2010-08-03T18:17:04.427 回答