1

LINQ CompiledQueryで動的なORDERBYを作成するにはどうすればよいですか(たとえば、コンパイルされたクエリのパラメーターとしてOrder FieldとDirectionを指定します)?

4

2 に答える 2

1

最初に本当に必要なのは、オブジェクトの文字列でプロパティ値にアクセスする方法だけです。リフレクションを使用できますが、遅いです。したがって、 http://stefan.rusek.org/Posts/LINQ-Expressions-as-Fast-Reflection-Invoke/3/のテストに基づくこのヘルパー クラス アプローチを使用します。

    public static class LINQHelper
    {
        public static IComparable OrderByProperty<TClass>(TClass item, 
                                                          string propertyName)
        {
            var t = Expression.Parameter(typeof(TClass), "t");
            var prop = Expression.Property(t, propertyName);
            var exp = Expression.Lambda(prop, t).Compile();
            return (IComparable)exp.DynamicInvoke(item);
        }

     }

プロパティ名の文字列による順序付けが必要なコードでは、この例の col1 では、次のことを行うだけです。

     var myQuery = from i in Items
                   select i;

     myQuery.OrderBy(i=>LINQHelper.OrderByProperty(i,"col1"));

お役に立てれば。

于 2011-11-23T15:46:39.167 に答える
1

私はそれを見つけたと思います:

このリンクをチェックしてください。以下の拡張メソッドを含むダイナミック Linq クエリ ライブラリを含む VS2008 コード サンプルを示します。これにより、次のことが可能になります。

Object.OrderBy("ColumnName");

拡張メソッドは次のとおりですが、ライブラリ全体が必要な場合もあります。

public static IQueryable<T> OrderBy<T>(this IQueryable<T> source, string ordering, params object[] values) {
    return (IQueryable<T>)OrderBy((IQueryable)source, ordering, values);
}

public static IQueryable OrderBy(this IQueryable source, 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)));
        methodAsc = "ThenBy";
        methodDesc = "ThenByDescending";
    }
    return source.Provider.CreateQuery(queryExpr);
}
于 2009-09-24T02:24:07.233 に答える