使用实体框架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中似乎没有起作用(我可能会做错事情.)
底线是我需要一个表达式树,以便基于用户输入动态地对视频集合进行排序.任何帮助将不胜感激.
解决方法
首先,您需要@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); } }
>列表< string>被转换为IQueryable< string>因为Enumerable运算符不采用表达式树.
>该方法以相反的顺序遍历排序列的列表(假设您要给列表中的第一个项目列出最高排序优先级)
.
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);