[英]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.