5

我知道已经提出了类似于这个问题的各种问题,但据我所知(和测试),所提供的解决方案似乎都不适合,所以就这样吧。

我想知道是否可以展平/非规范化对象层次结构,以便使用 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 的列表。

IE:

IEnumerable<DenormDistributionInfo> result = Mapper.Map(distributionInformationInstance);

使用 AutoMapper 是否可行/可行,还是我应该放弃并“手动”对其进行非规范化?

4

2 回答 2

4

主要的是,您希望避免在映射中“查找”源中未隐含的数据。“神奇”的映射会导致严重的维护问题。

然而,从概念上讲,这种映射非常简单。唯一复杂的因素是您需要两个源对象( aDistributionInformation和 a 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 执行此映射是否比手动执行更复杂或更简单,由您决定。在这种情况下,我认为我不会打扰,但在一个经常重复的更复杂的场景中,我可能会考虑它。

于 2015-05-05T10:08:57.263 回答
0

我对其进行了一些研究,虽然我选择通过“手动”映射来解决问题,但还有另一种方法(除了 Iain 发布的答案。虽然它确实感觉相当 hacky。

这个想法是使用类型转换器并将其映射两次

    public class DistributionInfoConverter : ITypeConverter<DistributionInformation, IEnumerable<DenormDistributionInfo>>
    {
        public IEnumerable<DenormDistributionInfo> Convert(ResolutionContext context)
        {
            var result = new List<DenormDistributionInfo>();
            var source = (DistributionInformation)context.SourceValue;

            foreach (var routeDetail in source.Routes)
            {
                var model = new DenormDistributionInfo();
                Mapper.Map(routeDetail, model);
                Mapper.Map(source, model);
                result.Add(model);
            }

            return result;
        }
    }

    Mapper.CreateMap<RouteInformation, DenormDistributionInfo>();
    Mapper.CreateMap<DistributionInformation, DenormDistributionInfo>()
    Mapper.CreateMap<DistributionInformation, IEnumerable<DenormDistributionInfo>>().ConvertUsing<DistributionInfoConverter>();

唯一的问题是,对于 DistributionInformation 的集合,您必须循环/选择每个项目并进行映射,而不是让 automapper 像往常一样让 automapper 弄清楚如何将集合映射到集合。

于 2015-05-06T07:35:00.067 回答