AutoMapper 展平嵌套集合

问题描述 投票:0回答:5

我尝试弄清楚如何将商家集合展平,每个商家都包含订单集合到 OrderViewModel 的平面列表。

这是我的 DTO:

public class Merchant
{
    public string MerchantName { get; set; }
    public List<Order> Orders { get; set; }
}

public class Order
{
    public string OrderId { get; set; }
}

这是视图模型:

public class OrderViewModel
{
    public string MerchantName { get; set; }
    public string OrderId { get; set; }
}

我的目标是将 List 扁平化为 List,而以下测试结构应产生 6 个视图模型:

var myMerchants = new List<Merchant>
{
    new Merchant
    {
        MerchantName = "Merchant X",
        Orders = new List<Order>
        {
             new Order { OrderId = "Order 1"},
             new Order { OrderId = "Order 2"},
             new Order { OrderId = "Order 3"}
        }
    },
    new Merchant
    {
        MerchantName = "Merchant Y",
        Orders = new List<Order>
        {
            new Order { OrderId = "Order 4"},
            new Order { OrderId = "Order 5"},
            new Order { OrderId = "Order 6"}
        }
    }
 };

 var models = Mapper.Map<List<OrderViewModel>>(myMerchants);
c# automapper
5个回答
10
投票

因为根对象的基数不是 1:1(即 2 个根

Merchants
需要映射到 6
OrderViewModels
),您可能需要求助于 自定义
TypeConverter
并在集合上进行操作级别,您可以使用
.SelectMany
进行展平:

public class MyTypeConverter : ITypeConverter<IEnumerable<Merchant>, List<OrderViewModel>>
{
    public List<OrderViewModel> Convert(ResolutionContext context)
    {
        if (context == null || context.IsSourceValueNull)
            return null;

        var source = context.SourceValue as IEnumerable<Merchant>;

        return source
            .SelectMany(s => s.Orders
              .Select(o => new OrderViewModel
              {
                  MerchantName = s.MerchantName,
                  OrderId = o.OrderId
              }))
              .ToList();
    }
}

然后您可以引导:

Mapper.CreateMap<IEnumerable<Merchant>, List<OrderViewModel>>()
    .ConvertUsing<MyTypeConverter>();

然后这样映射:

var models = Mapper.Map<List<OrderViewModel>>(myMerchants);

4
投票

一个有趣的发现是,只需执行以下操作就足以在没有自动映射器的情况下实现目标。

var models = myMerchants.SelectMany(s => s.Orders.Select(o => new OrderViewModel { MerchantName = s.MerchantName, OrderId = o.OrderId })).ToList();

1
投票

老问题,但认为对新版本会有帮助。

我正在使用 .Net core 2 和自动映射器。我更喜欢做可查询的 ProjectTo 扩展

queryable
   .SelectMany(outterClass => outterClass.innerList)
   .AsQueryable()
   .ProjectTo<OutterClassDto>();

然后,像这样配置:

config.CreateMap<OuterClass, OuterClassDto>();

0
投票

还可以将 SelectMany 与 AutoMapper 结合起来,并为扁平化集合中的每个项目进行映射。您需要此 Merchant -> OrderViewModel 和 Order -> OrderViewModel 的两个映射

var models = myMerchants.SelectMany(s => s.Orders.Select(o => 
{
    var mappedItem = Mapper.Map<OrderViewModel>(s);
    Mapper.Map(o, mappedItem);
    return mappedItem;
})).ToList();

0
投票

Automapper ConvertUsing() 也接受 Func。

 CreateMap<IEnumerable<Merchant>, IEnumerable<OrderViewModel>>()
     .ConvertUsing((src, desc) => src.SelectMany(m => m.Orders,
     (m, o) => new OrderViewModel
     {
         m.MerchantName,
         o.OrderId
         
     }).ToList());
© www.soinside.com 2019 - 2024. All rights reserved.