我在VS2008动态LINQ示例中找到了一个示例,允许您使用类似sql的字符串(例如OrderBy(“Name, Age DESC”))进行排序。不幸的是,所包含的方法只适用于IQueryable<T>。有什么办法得到这个功能IEnumerable<T>?
当前回答
我试图这样做,但有问题与Kjetil Watnedal的解决方案,因为我不使用内联linq语法-我更喜欢方法风格的语法。我的具体问题是尝试使用自定义IComparer进行动态排序。
我的解决方案是这样的:
给定一个IQueryable查询,如下:
List<DATA__Security__Team> teams = TeamManager.GetTeams();
var query = teams.Where(team => team.ID < 10).AsQueryable();
给定一个运行时排序字段参数:
string SortField; // Set at run-time to "Name"
动态OrderBy看起来是这样的:
query = query.OrderBy(item => item.GetReflectedPropertyValue(SortField));
这是使用一个叫做GetReflectedPropertyValue()的小助手方法:
public static string GetReflectedPropertyValue(this object subject, string field)
{
object reflectedValue = subject.GetType().GetProperty(field).GetValue(subject, null);
return reflectedValue != null ? reflectedValue.ToString() : "";
}
最后一件事——我提到过我想让OrderBy使用自定义iccomparer——因为我想做自然排序。
要做到这一点,我只需要改变OrderBy为:
query = query.OrderBy(item => item.GetReflectedPropertyValue(SortField), new NaturalSortComparer<string>());
有关NaturalSortComparer()的代码,请参阅这篇文章。
其他回答
感谢Maarten(在LINQ中使用PropertyInfo对象查询一个集合),我得到了这个解决方案:
myList.OrderByDescending(x => myPropertyInfo.GetValue(x, null)).ToList();
在我的情况下,我在一个“ColumnHeaderMouseClick”(WindowsForm)上工作,所以只发现特定的列按下及其对应的PropertyInfo:
foreach (PropertyInfo column in (new Process()).GetType().GetProperties())
{
if (column.Name == dgvProcessList.Columns[e.ColumnIndex].Name)
{}
}
OR
PropertyInfo column = (new Process()).GetType().GetProperties().Where(x => x.Name == dgvProcessList.Columns[e.ColumnIndex].Name).First();
(确保你的列名称与对象属性匹配)
干杯
你可以用这个:
public List<Book> Books(string orderField, bool desc, int skip, int take)
{
var propertyInfo = typeof(Book).GetProperty(orderField);
return _context.Books
.Where(...)
.OrderBy(p => !desc ? propertyInfo.GetValue(p, null) : 0)
.ThenByDescending(p => desc ? propertyInfo.GetValue(p, null) : 0)
.Skip(skip)
.Take(take)
.ToList();
}
我可以用下面的代码做到这一点。不需要编写长而复杂的代码。
protected void sort_array(string field_name, string asc_desc)
{
objArrayList= Sort(objArrayList, field_name, asc_desc);
}
protected List<ArrayType> Sort(List<ArrayType> input, string property, string asc_desc)
{
if (asc_desc == "ASC")
{
return input.OrderBy(p => p.GetType()
.GetProperty(property)
.GetValue(p, null)).ToList();
}
else
{
return input.OrderByDescending(p => p.GetType()
.GetProperty(property)
.GetValue(p, null)).ToList();
}
}
你可以将IEnumerable转换为IQueryable。
items = items.AsQueryable().OrderBy("Name ASC");
我试图这样做,但有问题与Kjetil Watnedal的解决方案,因为我不使用内联linq语法-我更喜欢方法风格的语法。我的具体问题是尝试使用自定义IComparer进行动态排序。
我的解决方案是这样的:
给定一个IQueryable查询,如下:
List<DATA__Security__Team> teams = TeamManager.GetTeams();
var query = teams.Where(team => team.ID < 10).AsQueryable();
给定一个运行时排序字段参数:
string SortField; // Set at run-time to "Name"
动态OrderBy看起来是这样的:
query = query.OrderBy(item => item.GetReflectedPropertyValue(SortField));
这是使用一个叫做GetReflectedPropertyValue()的小助手方法:
public static string GetReflectedPropertyValue(this object subject, string field)
{
object reflectedValue = subject.GetType().GetProperty(field).GetValue(subject, null);
return reflectedValue != null ? reflectedValue.ToString() : "";
}
最后一件事——我提到过我想让OrderBy使用自定义iccomparer——因为我想做自然排序。
要做到这一点,我只需要改变OrderBy为:
query = query.OrderBy(item => item.GetReflectedPropertyValue(SortField), new NaturalSortComparer<string>());
有关NaturalSortComparer()的代码,请参阅这篇文章。