AutoMapper 展平嵌套集合
AutoMapper flatten nested collections
我想弄清楚如何将商家集合扁平化,每个商家都包含一个订单集合到一个扁平的 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);
因为根对象的基数不是 1:1,(即 2 个根 Merchants
需要映射到 6 个 OrderViewModels
),您可能需要求助于 custom 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();
}
}
然后你可以bootstrap:
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();
老问题,但认为对新版本有帮助。
我正在使用 .Net core 2 和 automapper。我更喜欢做 queryable
的 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();
我想弄清楚如何将商家集合扁平化,每个商家都包含一个订单集合到一个扁平的 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
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
),您可能需要求助于 custom 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();
}
}
然后你可以bootstrap:
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();
老问题,但认为对新版本有帮助。
我正在使用 .Net core 2 和 automapper。我更喜欢做 queryable
的 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();