簡體   English   中英

AutoMapper 展平嵌套集合

[英]AutoMapper flatten nested collections

我試圖弄清楚如何將一個商家集合展平,每個商家都包含一個訂單集合到一個訂單視圖模型的平面列表。

這是我的 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<Merchant>展平為List< OrderViewModel>而以下測試結構應產生 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);

因為根對象的基數不是 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);

一個有趣的發現是,在沒有 automapper 的情況下,僅執行以下操作就足以實現目標。

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

舊問題,但認為對較新版本會有所幫助。

我正在使用帶有 automapper 的 .Net core 2。 我更喜歡做可查詢的 ProjectTo 擴展

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

然后,像這樣配置:

config.CreateMap<OuterClass, OuterClassDto>();

也可以將 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();

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM