gpt4 book ai didi

c# - 如何创建用于运行时排序的表达式树?

转载 作者:可可西里 更新时间:2023-11-01 08:39:46 25 4
gpt4 key购买 nike

使用 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 中不起作用。(我可能做错了什么。)

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

最佳答案

首先你需要OrderBy @Slace 写的扩展方法 here .所有归功于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);

关于c# - 如何创建用于运行时排序的表达式树?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11336713/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com