如何创建运行时表达式树排序?(How do I create an expression tree

2019-06-26 01:58发布

使用实体框架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列举建立在运行时表达式树。 此外,用户可以指定升序或降序排序和表达式树要么处理。

我试图动态LINQ库VS 2008,但它不会出现在VS 2010的工作(我可能做错了什么。)

底线是,我需要一个表达式树动态排序基于用户输入的视频采集。 任何帮助,将不胜感激。

Answer 1:

首先,你需要的OrderBy是@Slace写扩展方法在这里 。 所有信贷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. 通过以相反的顺序排序列的列表中的迭代方法(假设你想给的第一个项目在列表中的最高优先级排序)

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);


Answer 2:

这正是我需要的凯文。 我注意到的是,如果你使用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);
    }


文章来源: How do I create an expression tree for run time sorting?