当我尝试映射对象的空列表(成员)时出现问题,考虑到我指定了:
.ForAllMembers(opts => opts.Condition((src, dest, srcMember) =>
srcMember != null
));
cfg.AllowNullCollections = true; // didn't help also
代码中的简短示例:
gi.PersonList = new List<Person>();
gi.PersonList.Add(new Person { Num = 1, Name = "John", Surname = "Scott" });
GeneralInfo gi2 = new GeneralInfo();
gi2.Qty = 3;
Mapper.Map<GeneralInfo, GeneralInfo>(gi2, gi);
gi.PersonList.Count = 0
,如何解决这个问题?
using System;
using System.Collections.Generic;
using AutoMapper;
public class Program
{
public static void Main(string[] args)
{
Mapper.Initialize(cfg =>
{
cfg.AllowNullCollections = true;
cfg.CreateMap<GeneralInfo, GeneralInfo>()
.ForAllMembers(opts => opts.Condition((src, dest, srcMember) =>
srcMember != null
));
});
GeneralInfo gi = new GeneralInfo();
gi.Descr = "Test";
gi.Dt = DateTime.Now;
gi.Qty = 1;
gi.PersonList = new List<Person>();
gi.PersonList.Add(new Person { Num = 1, Name = "John", Surname = "Scott" });
GeneralInfo gi2 = new GeneralInfo();
gi2.Qty = 3;
Console.WriteLine("Count antes de mapeo = " + gi.PersonList.Count);
Mapper.Map<GeneralInfo, GeneralInfo>(gi2, gi);
Console.WriteLine("Count despues de mapeo = " + gi.PersonList.Count);
// Error : gi.PersonList.Count == 0 !!!!
//por que? si arriba esta: Condition((src, dest, srcMember) => srcMember != null ...
}
}
class Person
{
public int Num { get; set; }
public string Name { get; set; }
public string Surname { get; set; }
}
class GeneralInfo
{
public int? Qty { get; set; }
public DateTime? Dt { get; set; }
public string Descr { get; set; }
public List<Person> PersonList { get; set; }
}
答案 0 :(得分:5)
这应该可行,但我不确定你是否想要像这样微观管理:
cfg.AllowNullCollections = true;
cfg.CreateMap<GeneralInfo, GeneralInfo>()
.ForMember(x => x.PersonList, opts => opts.PreCondition((src) => src.PersonList != null));
问题是专门处理的集合(对于大多数映射器来说都是如此,虽然AutoMapper在这种情况下有点奇怪,但它不是我最喜欢的)并且似乎需要初始化目标集合。正如我所看到的,集合不是完全复制的,但是你需要初始化和复制单个项目(这是我的演绎,但听起来不错)。
即。即使您跳过源,目标仍将最终重新初始化(空)。
问题似乎是Condition
,在给定documentation时,在稍后的某个时间点应用,此时目标已经初始化。
PreCondition
有一个不同的签名可以像你想象的那样使用,因为它没有实际值,只有源可用。
似乎唯一有用的解决方案是使用“每个成员”PreCondition
(如上所述)。
修改强>:
......或者这个(使用ForAllMembers
),但有点难看,反思等。
cfg.CreateMap<GeneralInfo, GeneralInfo>()
.ForAllMembers(opts =>
{
opts.PreCondition((src, context) =>
{
// we can do this as you have a mapping in between the same types and no special handling
// (i.e. destination member is the same as the source property)
var property = opts.DestinationMember as System.Reflection.PropertyInfo;
if (property == null) throw new InvalidOperationException();
var value = property.GetValue(src);
return value != null;
});
}
);
...但似乎没有任何更清洁的支持。
编辑(BUG&amp; FINAL THOUGHTS):
Conditional mapping to existing collection doesn't work from version 5.2.0 #1918
正如评论中指出的那样(@LucianBargaoanu),这似乎是一个真正的错误,因为它在这个“角落”的情况下是不一致的(尽管我不同意这一点,这是一个非常典型的场景)收藏和传递目的地。在这种情况下,它几乎使Condition
无效,因为目标已经初始化/清除。
唯一的解决方案似乎确实是PreCondition
(但鉴于签名不同,它有问题,我个人不确定为什么他们没有将同样多的参数传递到PreCondition中井吗)。
还有更多信息:
nest collection is clear when using Condition but not Ignore #1940
Collection property on destination object is overwritten despite Condition() returning false #2111
Null source collection emptying destination collection #2031
答案 1 :(得分:1)
试试这个;
gi = Mapper.Map<GeneralInfo, GeneralInfo>(gi2);
我最近遇到了这个问题,如果目的地和来源是同一类型,目的地参数不能按预期工作。
另外,我想通知它与Collection
个对象无关。如果您调试gi
对象,您将看到其他属性也保留旧值。如果您将目标实例作为destionation参数传递,那么Automapper
不会更改之前分配的属性值。我认为,Automapper
并非设计用于创建复制/克隆对象。