2013-01-08 34 views
3

我遇到了Automapper问题。我建立了一个测试窗体表单应用程序,下面是代码。也看看每个MessageBox后的意见:Automapper用子对象覆盖丢失的源属性列表

public class FirstClass 
    { 
     public string FirstProp { get; set; } 
     public IList<FirstClassChild> Children { get; set; } 
    } 

    public class FirstClassChild 
    { 
     public string FirstChildProp { get; set; } 
    } 

    public class SecondClass 
    { 
     public string FirstProp { get; set; } 
     public string SecondProp { get; set; } 
     public IList<SecondClassChild> Children { get; set; } 
    } 

    public class SecondClassChild 
    { 
     public string FirstChildProp { get; set; } 
     public string SecondChildProp { get; set; } 
    } 

    public partial class Form1 : Form 
    { 
     public Form1() 
     { 
      InitializeComponent(); 

      AutoMapper.Mapper.CreateMap<FirstClass, SecondClass>(); 
      AutoMapper.Mapper.CreateMap<FirstClassChild, SecondClassChild>(); 

      var f = new FirstClass { FirstProp = "FirstClass" }; 
      f.Children = new List<FirstClassChild> { new FirstClassChild { FirstChildProp = "FirstClass" } }; 
      var s = new SecondClass { FirstProp = "SecondClass", SecondProp = "SecondClass" }; 
      s.Children = new List<SecondClassChild> { new SecondClassChild { FirstChildProp = "SecondClass", SecondChildProp = "SecondClass" } }; 
      AutoMapper.Mapper.Map(f, s); 

      var fc = new FirstClassChild { FirstChildProp = "FirstClass" }; 
      var sc = new SecondClassChild { FirstChildProp = "SecondClass", SecondChildProp = "SecondClass" }; 
      AutoMapper.Mapper.Map(fc, sc); 

      MessageBox.Show(sc.FirstChildProp);//FirstClass as expected 
      MessageBox.Show(sc.SecondChildProp);//SecondClass as expected 

      MessageBox.Show(s.FirstProp);//FirstClass as expected 
      MessageBox.Show(s.SecondProp);//SecondClass as expected 
      MessageBox.Show(s.Children.First().FirstChildProp);//FirstClass as expected 
      MessageBox.Show(s.Children.First().SecondChildProp);//Empty not expected!! 

     } 
    } 

我该怎么做才能避免这种情况?预期这种行为? 无论如何,任何人都可以指导我如何使SecondClass孩子SecondChildProp保持“SecondClass”,因为它是在映射发生之前。

回答

5

我问了一个类似的问题here,发现另一个类似的问题here

我认为@PatrickSteele提出了一个很好的观点:AutoMapper应该如何将源列表映射到现有对象的目标列表,当dest列表可能不一定与源列表有任何相似之处?即“But what if one list has 3 and the other list has 5?

如果你确信FirstClassSecondClass有相同数量的Children如果FirstClass的第n个子总是对应于SecondClass的第N个孩子,你可以尝试这样的事情:

Mapper.CreateMap<FirstClass, SecondClass>() 
    .ForMember(m => m.Children, o => o.Ignore()) 
    .AfterMap((src, dest) => 
     { 
      for (var i = 0; i < dest.Children.Count; i++) 
       Mapper.Map(src.Children[i], dest.Children[i]); 
     }); 

,或者如果FirstChildProp是某种独特的键:

Mapper.CreateMap<FirstClass, SecondClass>() 
    .ForMember(m => m.Children, o => o.Ignore()) 
    .AfterMap((src, dest) => 
     { 
      foreach (var dChild in dest.Children) 
      { 
       var sChild = src.Children.Single(c => c.FirstChildProp == dChild.FirstChildProp); 
       Mapper.Map(sChild, dChild); 
      } 
     }); 
+0

这很有道理,谢谢。我不承认复杂性,被我孤立的案件蒙上了一层阴影。 – John

+0

第二个想法,关于AutoMapper无法知道列表中的子项的论证等等。AutoMapper映射FirstChildProp是否有点尴尬?似乎与我不一致。 – John

+1

@John - 在幕后,我不认为Automapper将FirstChildProp **映射到**现有的SecondClassChild对象中。相反,我期望它完全** **了SecondClass的'Children'列表,并用它刚刚创建的新的SecondClassChild列表覆盖它(使用'FirstClass.Children'作为映射源)。这应该变得更加明显当source.Children和dest.Children列表具有不同的长度时。 – Merenzo