如何为运行时排序创建表达式树?

时间:2012-07-05 00:49:04

标签: c# entity-framework-4 lambda

使用Entity Framework 4,我正在尝试基于成员名称集合实现动态排序。基本上,用户可以选择要排序的字段和排序顺序。我看过表达式树的例子,不能把它拼凑在一起。以下是一些细节:

列名集合:

public List<string> sortColumns;
sortColumns = new List<string>();

/// Example subset of video fields.  The collection will vary.
sortColumns.Add("Width");
sortColumns.Add("Height");
sortColumns.Add("Duration");
sortColumns.Add("Title");

视频类定义如下:

public class Video
{
    public string Title { get; set; }
    public int Width { get; set; }
    public int Height { get; set; }
    public float Duration { get; set; }
    public string Filename { get; set; }
    public DateTime DateCreated { get; set; }
    .
    .
    .
}
public List<Video> Videos;

我想要做的是通过sortColumns集合枚举以在运行时构建表达式树。此外,用户可以指定升序或降序排序,表达式树应该处理。

我为VS 2008尝试了动态LINQ库,但它似乎在VS 2010中不起作用。(我可能做错了。)

底线是我需要一个表达式树来根据用户输入动态地对视频集合进行排序。任何帮助将不胜感激。

2 个答案:

答案 0 :(得分:17)

首先,您需要@Slace编写hereOrderBy扩展方法。所有信用都归Slace所有,这是一段很棒的代码,也是解决方案中最困难的部分!我对它进行了一些修改,以适应您的具体情况:

public static class QueryableExtensions
{
    public static IQueryable<T> OrderBy<T>(this IQueryable<T> source, string sortProperty, ListSortDirection sortOrder)
    {
        var type = typeof(T);
        var property = type.GetProperty(sortProperty);
        var parameter = Expression.Parameter(type, "p");
        var propertyAccess = Expression.MakeMemberAccess(parameter, property);
        var orderByExp = Expression.Lambda(propertyAccess, parameter);
        var typeArguments = new Type[] { type, property.PropertyType };
        var methodName = sortOrder == ListSortDirection.Ascending ? "OrderBy" : "OrderByDescending";
        var resultExp = Expression.Call(typeof(Queryable), methodName, typeArguments, source.Expression, Expression.Quote(orderByExp));

        return source.Provider.CreateQuery<T>(resultExp);
    }
}

创建一个方法来对列表进行排序。以下方法需要注意以下几点:

  1. List<string>转换为IQueryable<string>,因为Enumerable运算符不接受表达式树。
  2. 该方法以相反的顺序遍历排序列列表(假设您要为列表中的第一项提供最高排序优先级)
  3. private void PrintVideoList(IEnumerable<string> sortColumns, ListSortDirection sortOrder)
    {
        var videos = this.GetVideos();
        var sortedVideos = videos.AsQueryable();
    
        foreach (var sortColumn in sortColumns.Reverse())
        {
            sortedVideos = sortedVideos.OrderBy(sortColumn, sortOrder);
        }
    
        // Test the results
        foreach (var video in sortedVideos)
        {
            Console.WriteLine(video.Title);
        }
    }
    

    然后你应该能够使用这样的方法:

    // These values are entered by the user
    var sortColumns = new List<string> { "Width", "Title", "Height" };
    var sortOrder = ListSortDirection.Ascending;
    
    // Print the video list base on the user selection
    this.PrintVideoList(sortColumns, sortOrder);
    

答案 1 :(得分:0)

正是我需要凯文。我注意到的是,如果你使用orderby它只需要最后的orderby选择。

我添加了这个方法(ThenBy)到我的,它似乎运作良好

public static IQueryable<T> ThenBy<T>(this IQueryable<T> source, string sortProperty, ListSortDirection sortOrder)
    {
        var type = typeof(T);
        var property = type.GetTypeInfo().GetDeclaredProperty(sortProperty);
        var parameter = Expression.Parameter(type, "p");
        var propertyAccess = Expression.MakeMemberAccess(parameter, property);
        var orderByExp = Expression.Lambda(propertyAccess, parameter);
        var typeArguments = new Type[] { type, property.PropertyType };
        var methodName = sortOrder == ListSortDirection.Ascending ? "ThenBy" : "ThenByDescending";
        var resultExp = Expression.Call(typeof(Queryable), methodName, typeArguments, source.Expression, Expression.Quote(orderByExp));

        return source.Provider.CreateQuery<T>(resultExp);
    }