タイトルが説明するように、複数の並べ替えパラメーターのサポートを含め、動的Linqを使用して自然な並べ替えを実現する方法はありますか?
好ましくは私はこのようなことをしたいと思います(カスタムIComparerを使用して):
List<Invoice> invoices = Provider.GetInvoices();
invoices = invoices
.AsQueryable()
.OrderBy("SortingParameter1 ASC, SortingParamaeter 2 ASC", new NaturalSort())
.ToList();
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などの他のプロバイダーで機能するかどうかはわかりません