LINQ to Entities select 子句中的可重用 functions/expressions

Reusable functions/expressions in LINQ to Entities select clauses

我有很多将实体转换为数据传输对象的代码,与此类似:

var result = from t in DatabaseContext.SomeTable
             where t.Value1 = someValue
             select new SomeTableDTO()
             {
                  Value1 = t.Value1,
                  Value2 = t.Value2,
                  SomeParent = new SomeParentDTO()
                  {
                      Value3 = t.SomeParent.Value3,
                      Value4 = t.SomeParent.Value4
                  }
              };

这行得通,但问题是我一遍又一遍地重复相同的代码,因为有很多 DTO 都有 SomeParent 属性。

var result = from t in DatabaseContext.SomeOtherTable
             where t.Value5 = someValue
             select new SomeOtherTableDTO()
             {
                  Value5 = t.Value5,
                  Value6 = t.Value6,
                  SomeParent = new SomeParentDTO()
                  {
                      Value3 = t.SomeParent.Value3,
                      Value4 = t.SomeParent.Value4
                  }
              };

我想做这样的事情,以便可以共享 SomeParentDTO 的转换:

var result = from t in DatabaseContext.SomeTable
             where t.Value1 = someValue
             select new SomeTableDTO()
             {
                  Value1 = t.Value1,
                  Value2 = t.Value2,
                  SomeParent = SomeParentConverter(t.SomeParent)
              };

.

Func<SomeParent, SomeParentDTO> SomeParentConverter = (parent) =>
{
    return new SomeParentDTO()
    {
        Value3 = parent.Value3,
        Value4 = parent.Value4
    };
};

但这当然行不通,因为 Invoke 在 LINQ to Entities 中不受支持。 This posting 似乎朝着我想要的方向发展,但它使用了一个传递给 .Select() 的单个 Expression,这并没有解决我的 DRY 问题。

有什么方法可以完成我想要的吗?

您可以使用 Automapper. It supports mappings in IQueryable extensions.

文档中的部分示例:

public List<OrderLineDTO> GetLinesForOrder(int orderId)
{
    Mapper.CreateMap<OrderLine, OrderLineDTO>()
        .ForMember(dto => dto.Item, conf => conf.MapFrom(ol => ol.Item.Name);

    using (var context = new orderEntities())
    {
        return context.OrderLines.Where(ol => ol.OrderId == orderId)
                    .Project().To<OrderLineDTO>().ToList();
    }
}