2011-11-23 120 views
7

我想我錯過了一個簡單的概念與valueinjecter和/或AutoMapper,但你如何深入克隆父dto.Entity到biz.Entity幷包括所有的孩子?例如,biz.person.InjectFrom(dto.person)。我想將dto.person.AddressList集合複製到biz.person.AddressList集合中,儘管dto.Addressbiz.Address與類型不同,但具有相同的屬性名稱。omu.valueinjecter深入克隆不像類型

我的想法是,如果Parent屬性名拼寫相同,例如AddressList,那麼2個底層對象是不同的類型都沒關係。它仍然會複製相同名稱的簡單類型如int,字符串等

謝謝

+0

你看着從ValueInjecter的CodePlex網站頁面的深層克隆頁面? http://valueinjecter.codeplex.com/wikipage?title=Deep%20Cloning&referringTitle=Home – Omu

+0

嘿Chuck。是的,我做到了。它並沒有深入地克隆我的[nhibernate]子實體 – user52212

+0

如果你發佈一些代碼並且關於不同類型,默認InjectFrom()從相同名稱和相同類型注入將會很好,所以它不會影響成員不同的類型(如果你正在從一種類型轉換到另一種類型,那麼這就是爲什麼DeepClone不能爲你工作) – Omu

回答

7

我有同樣的問題時,陣列/中的對象列表中具有相同的名稱,但不同類型的(即名爲Animal的ORMAnimals []類型的屬性映射到一個名爲Animals的屬性Animals [])。

有了一些小的調整,示例代碼查克·諾里斯對深克隆page我得到了它在我的測試代碼的工作:

public class CloneInjection : ConventionInjection 
{ 
    protected override bool Match(ConventionInfo c) 
    { 
     return c.SourceProp.Name == c.TargetProp.Name && c.SourceProp.Value != null; 
    } 

    protected override object SetValue(ConventionInfo c) 
    { 
     //for value types and string just return the value as is 
     if (c.SourceProp.Type.IsValueType || c.SourceProp.Type == typeof(string) 
      || c.TargetProp.Type.IsValueType || c.TargetProp.Type == typeof(string)) 
      return c.SourceProp.Value; 

     //handle arrays 
     if (c.SourceProp.Type.IsArray) 
     { 
      var arr = c.SourceProp.Value as Array; 
      var clone = Activator.CreateInstance(c.TargetProp.Type, arr.Length) as Array; 

      for (int index = 0; index < arr.Length; index++) 
      { 
       var a = arr.GetValue(index); 
       if (a.GetType().IsValueType || a.GetType() == typeof(string)) continue; 
       clone.SetValue(Activator.CreateInstance(c.TargetProp.Type.GetElementType()).InjectFrom<CloneInjection>(a), index); 
      } 
      return clone; 
     } 


     if (c.SourceProp.Type.IsGenericType) 
     { 
      //handle IEnumerable<> also ICollection<> IList<> List<> 
      if (c.SourceProp.Type.GetGenericTypeDefinition().GetInterfaces().Contains(typeof(IEnumerable))) 
      { 
       var t = c.TargetProp.Type.GetGenericArguments()[0]; 
       if (t.IsValueType || t == typeof(string)) return c.SourceProp.Value; 

       var tlist = typeof(List<>).MakeGenericType(t); 
       var list = Activator.CreateInstance(tlist); 

       var addMethod = tlist.GetMethod("Add"); 
       foreach (var o in c.SourceProp.Value as IEnumerable) 
       { 
        var e = Activator.CreateInstance(t).InjectFrom<CloneInjection>(o); 
        addMethod.Invoke(list, new[] { e }); // in 4.0 you can use dynamic and just do list.Add(e); 
       } 
       return list; 
      } 

      //unhandled generic type, you could also return null or throw 
      return c.SourceProp.Value; 
     } 

     //for simple object types create a new instace and apply the clone injection on it 
     return Activator.CreateInstance(c.TargetProp.Type) 
      .InjectFrom<CloneInjection>(c.SourceProp.Value); 
    } 
} 
+0

addMethod.Invoke(list,new [] {e}); //在4.0中,你可以使用動態,只需做list.Add(e);有沒有人做過這個?我試着用'動態列表'替換'var list',然後做list.Add(e),它編譯但引發了一個運行時異常。 –

+0

@Gloopy - 嘗試了這個,但我一直得到堆棧溢出:( – Ryan

+0

@Ryan是否有可能你的對象有一個引用回到自己或另一個父對象導致無限循環,而遍歷孩子? – Gloopy