我知道有人提出了与这个问题类似的各种问题,但据我所知(和测试),所提供的解决方案似乎都不合适,所以这里开始吧。
我想知道是否可以扁平化/非规范化对象层次结构,以便使用 AutoMapper 将具有嵌套属性列表的实例映射到某个目标类型的列表。
我有一个看起来像这样的源类
来源:
public class DistributionInformation
{
public string Streetname;
public RouteInformation[] Routes;
}
public class RouteInformation
{
public int RouteNumber;
public string RouteDescription;
}
目的地:
public class DenormDistributionInfo
{
public string Streetname;
public int RouteNumber;
public string RouteDescription;
}
所以我想将这两个源映射到非规范化目标 DenormDistributionInfo 的列表。
即:
IEnumerable<DenormDistributionInfo> result = Mapper.Map(distributionInformationInstance);
使用 AutoMapper 是否可能/可行,或者我应该放弃并“手动”对其进行非规范化?
最佳答案
主要是您希望避免在映射中“查找”源中未隐含的数据。 “神奇”映射会导致严重的维护问题。
然而,从概念上讲,这种映射非常简单。唯一复杂的因素是您需要两个 源对象(DistributionInformation
和RouteInformation
)来构造您的目标对象。如果你遵循这个思路,我们可以创建一个非魔法映射,它清楚地保留了我们的意图——我会这样做:-
// We need both source objects in order to perform our map
Mapper.CreateMap<Tuple<DistributionInformation, RouteInformation>, DenormDistributionInfo>()
.ForMember(d => d.Streetname, o => o.MapFrom(s => s.Item1.Streetname))
.ForMember(d => d.RouteDescription, o => o.MapFrom(s => s.Item2.RouteDescription))
.ForMember(d => d.RouteNumber, o => o.MapFrom(s => s.Item2.RouteNumber));
// We can use ConstructUsing to pass both our source objects to our map
Mapper.CreateMap<DistributionInformation, IEnumerable<DenormDistributionInfo>>()
.ConstructUsing(
x => x.Routes
.Select(y => Mapper.Map<DenormDistributionInfo>(Tuple.Create(x, y)))
.ToList());
并调用它:-
var flattened = Mapper.Map<IEnumerable<DenormDistributionInfo>>(source);
如果您愿意,可以通过创建一个 DTO 来保存两个源对象来避免一点元组恐怖。如果您的实际代码比您在问题中提供的示例稍微复杂一些,我特别强烈建议您这样做。
是否使用 AutoMapper 执行此映射比手动完成更复杂或更简单取决于您的决定。在这种情况下,我认为我不会打扰,但在更复杂且经常重复的情况下,我可能会考虑。
关于c# - 使用自动映射器对对象层次结构进行非规范化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30049357/