【问题标题】:How to use a string to create a EF order by expression?如何使用字符串通过表达式创建 EF 订单?
【发布时间】:2017-02-15 22:21:03
【问题描述】:

我正在尝试实现此转换

"Address.Street" => (p) => p.Address.Street
"Name" => (p) => p.Name

我能够找到一种使用反射通过表达式生成订单的方法,但它不适用于像 Address.Street 这样的复杂排序,因为它适用于单个属性级别。

有没有办法做到这一点?我已经看到我编译了 lambda 表达式,但我不明白如何使它适用于这种情况。

【问题讨论】:

标签: c# entity-framework linq


【解决方案1】:

创建表达式并不难,但棘手的部分是如何在不知道属性类型的情况下将其绑定到对应的OrderBy(Descending) / ThenBy(Descendig) 方法(因此选择器表达式结果的类型)。

以下是封装在自定义扩展方法中的所有内容:

public static partial class QueryableExtensions
{
    public static IOrderedQueryable<T> OrderByMember<T>(this IQueryable<T> source, string memberPath)
    {
        return source.OrderByMemberUsing(memberPath, "OrderBy");
    }
    public static IOrderedQueryable<T> OrderByMemberDescending<T>(this IQueryable<T> source, string memberPath)
    {
        return source.OrderByMemberUsing(memberPath, "OrderByDescending");
    }
    public static IOrderedQueryable<T> ThenByMember<T>(this IOrderedQueryable<T> source, string memberPath)
    {
        return source.OrderByMemberUsing(memberPath, "ThenBy");
    }
    public static IOrderedQueryable<T> ThenByMemberDescending<T>(this IOrderedQueryable<T> source, string memberPath)
    {
        return source.OrderByMemberUsing(memberPath, "ThenByDescending");
    }
    private static IOrderedQueryable<T> OrderByMemberUsing<T>(this IQueryable<T> source, string memberPath, string method)
    {
        var parameter = Expression.Parameter(typeof(T), "item");
        var member = memberPath.Split('.')
            .Aggregate((Expression)parameter, Expression.PropertyOrField);
        var keySelector = Expression.Lambda(member, parameter);
        var methodCall = Expression.Call(
            typeof(Queryable), method, new[] { parameter.Type, member.Type },
            source.Expression, Expression.Quote(keySelector));
        return (IOrderedQueryable<T>)source.Provider.CreateQuery(methodCall);
    }

【讨论】:

  • 不客气,伙计 :) 你的话比任何虚拟积分都值钱!
  • 我也喜欢。是否也可以有一个 IEnumerable 版本...而客户端代码在调用方法之前总是必须调用 .AsQueryable() ?
  • @ManOfSteele 当然有可能。最简单的就是为IEnumerable&lt;T&gt;定义4个类似的方法,并用source.AsQueryable()调用对应的方法并转换结果。或者创建一个类似的实现,但不是 Expression.Call 编译 keySelector 并通过反射调用相应的 Enumerable 方法。
猜你喜欢
  • 2015-11-04
  • 2020-03-28
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-08-21
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多