自然排序与动态Linq,包括多个排序参数

本文关键字:排序 包括多 参数 Linq 动态 自然 | 更新日期: 2023-09-27 18:16:23

正如标题所描述的,是否有任何方法可以使用动态Linq实现自然排序,包括支持多个排序参数?

最好我想做这样的事情(使用自定义IComparer):

List<Invoice> invoices = Provider.GetInvoices();
invoices = invoices
  .AsQueryable()
  .OrderBy("SortingParameter1 ASC, SortingParamaeter 2 ASC", new NaturalSort())
  .ToList();

自然排序与动态Linq,包括多个排序参数

DynamicLinq没有将IComparer<T>作为参数的方法OrderBy,因此您不能传递自定义比较器,但您可以像这样修改源代码

public static IQueryable<T> OrderBy<T,ComparerType>(this IQueryable<T> source, IComparer<ComparerType> comparer, string ordering, params object[] values)
{
    return (IQueryable<T>)OrderBy((IQueryable)source, comparer, ordering, values);
}
public static IQueryable OrderBy<ComparerType>(this IQueryable source, IComparer<ComparerType> comparer, string ordering, params object[] values)
{
    if (source == null) throw new ArgumentNullException("source");
    if (ordering == null) throw new ArgumentNullException("ordering");
    ParameterExpression[] parameters = new ParameterExpression[] {
        Expression.Parameter(source.ElementType, "") };
    ExpressionParser parser = new ExpressionParser(parameters, ordering, values);
    IEnumerable<DynamicOrdering> orderings = parser.ParseOrdering();
    Expression queryExpr = source.Expression;
    string methodAsc = "OrderBy";
    string methodDesc = "OrderByDescending";
    foreach (DynamicOrdering o in orderings)
    {
        queryExpr = Expression.Call(
            typeof(Queryable), o.Ascending ? methodAsc : methodDesc,
            new Type[] { source.ElementType, o.Selector.Type },
            queryExpr, Expression.Quote(Expression.Lambda(o.Selector, parameters)), Expression.Constant(comparer));
            methodAsc = "ThenBy";
            methodDesc = "ThenByDescending";
        }
    return source.Provider.CreateQuery(queryExpr);
}

并像这样使用

List<Invoice> invoices = Provider.GetInvoices();
invoices = invoices.AsQueryable()
                   .OrderBy(new NaturalSort(), "SortingParameter1 ASC, SortingParamaeter 2 ASC")
                   .ToList();

其中NaturalSort应实现IComparer<T>关于自然排序的实现,你可以在c#中看到这个自然排序顺序

注意:但我不确定这将与其他提供商一起工作,如db

看起来你想要这样:

invoices = invoices
    .OrderBy(invoice => invoice.SortingParameter1, new NaturalSort())
    .ThenBy(invoice => invoice.SortingParameter2, new NaturalSort())
    .ToList();

基本上,您希望将OrderByOrderByDescending用于第一个排序参数,然后将ThenByThenByDescending用于其余参数。

(注意这里不需要AsQueryable,因为List<T>扩展了IEnumerable<T>)