我有一个对象图,我正在使用EF CodeFirst和AutoMapper从数据库加载到DTO中:
public class Foo { public int Id { get; set; } public virtual ICollection<Bar> Bars { get; set; } } public class Bar { public int Id { get; set; } public int FooId { get; set; } public virtual Foo Foo { get; set; } public string Name { get; set; } public int SortOrder { get; set; } } public class FooDto { public IEnumerable<BarDto> Bars { get; set; } } public class BarDto { public string Name { get; set; } public int SortOrder { get; set; } }
我的映射看起来像:
mapper.CreateMap<Foo,FooDto>(); mapper.CreateMap<Bar,BarDto>();
到现在为止还挺好.我可以从我的上下文和项目中获取实体到DTO:
var foos = context.Foos.Project().To<FooDto>();
然而,我无法使用这种方法,它是通过其在IQueryable中的SortOrder来对Bars进行排序的.
如果我尝试:
mapper.CreateMap<Foo,FooDto>() .ForMember( x => x.Bars opt => opt.MapFrom(src => src.Bars.OrderBy(x => x.SortOrder))); mapper.CreateMap<Bar,BarDto>(); var foos = context.Foos.Project().To<FooDto>();
我有一个例外:
System.InvalidOperationException: Sequence contains no elements at System.Linq.Enumerable.First[TSource](IEnumerable`1 source) at AutoMapper.MappingEngine.CreateMapExpression(Type typeIn,Type typeOut) ...
这似乎与https://github.com/AutoMapper/AutoMapper/issues/159有关 – 虽然我已经在使用一个复杂的类型作为子集合.我猜CreateMapExpression不支持OrderBy的子集合?
如果我没有使用.Project().To(),那么我可以轻松地排序孩子集合:
var model = context.Foos.Select(x => new FooDto() { Bars = x.Bars.OrderBy(y => y.SortOrder) });
但是我必须重复映射,无论我想要使用它,都击败了使用AutoMapper的目的.
奇怪的是: –
1)我可以对孩子集合执行其他(更复杂的)操作,并将这些操作放在我的父DTO中没有问题:
mapper.CreateMap<Foo,FooDto>() .ForMember( x => x.AllBarsHaveAName,opt => opt.MapFrom(src => src.Bars.All(x => x.Name != null)));
2)我可以Mapper.Map< FooDto>(foo);记忆力好,它会排序酒吧没问题.
可以在IQueryable级别对子集合进行排序,同时仍然使用.Project().To()?