【问题标题】:How do I create an expression tree for run time sorting?如何为运行时排序创建表达式树?
【发布时间】:2012-07-05 00:49:04
【问题描述】:

使用 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 的 Dynamic LINQ 库,但它似乎在 VS 2010 中不起作用。(我可能做错了什么。)

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

【问题讨论】:

    标签: c# entity-framework-4 lambda


    【解决方案1】:

    首先,您需要@Slace 编写的here 扩展方法OrderBy。所有功劳归功于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&lt;string&gt; 被转换为 IQueryable&lt;string&gt;,因为 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);
    

    【讨论】:

    • @KevinAenmey 此代码适用于视频对象列表。在我这边,我有嵌套类,所以要访问一些值,我有像“Event.Volume.Market_Leader_Volume”这样的标签,sortColumn 将具有价值。您的代码显然在这里不起作用,因为“属性”将为空,并且我在下一行“var propertyAccess = Expression.MakeMemberAccess(parameter, property);”上有一个空异常。有什么办法解决这个问题吗?
    【解决方案2】:

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

    【讨论】:

    • 我知道这是一篇旧文章,但您是否能够扩展它以使用嵌套(导航)属性?
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2014-03-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多