2

最終的にSQLのような結果になるContains式を動的に構築する式ツリーを変更しようとしています

P IN (123, 124, 125, 200, 201)

代わりにチェックして範囲チェックを実行すると、最終的に次のようなSQLになります

(P >= 123 AND P <= 125) OR (P >= 200 AND P <= 201)

この投稿に基づいてソリューションを作成しています。

static public Expression<Func<TElement, bool>> 
BuildContainsExpression<TElement, TValue>(
    Expression<Func<TElement, TValue>> valueSelector, IEnumerable<TValue> values)
{
    // Removed for post: Input checks and edge cases

    var equals = 
      values.Select(value => 
       (Expression)Expression.Equal(valueSelector.Body, 
                                    Expression.Constant(value, typeof(TValue))));

    var body = equals.Aggregate<Expression>((accumulate, equal) => 
                                             Expression.Or(accumulate, equal));

    return Expression.Lambda<Func<TElement, bool>>(body, p);
}

比較用の値を指定すると、範囲チェックを機能させることができます。

long testValue = 5;
List<KeyValuePair<int, int>> ranges = new List<KeyValuePair<int, int>>() 
{
    new KeyValuePair<long, long>(3, 6),
    new KeyValuePair<long, long>(10, 12),
    new KeyValuePair<long, long>(20, 20),
};

List<BinaryExpression> rangeExpressions = new List<BinaryExpression>();

foreach (var pair in ranges)
{
    var greaterThanOrEqual = 
        Expression.GreaterThanOrEqual(Expression.Constant(testValue), 
                                      Expression.Constant(pair.Key));

    var lessThanOrEqual = 
        Expression.LessThanOrEqual(Expression.Constant(testValue), 
                                   Expression.Constant(pair.Value));

    var inRange = Expression.AndAlso(greaterThanOrEqual, lessThanOrEqual);
    rangeExpressions.Add(inRange);
}

var final = 
    rangeExpressions.Aggregate<Expression>((a, b) => Expression.Or(a, b));

var result = Expression.Lambda<Func<bool>>(final).Compile()();

しかし、そのコードを Linq で使用するメソッドにドロップすると、渡された式から比較用の値を取得する方法がわかりません。そのメソッドのシグネチャは次のとおりです。

Expression<Func<TElement, bool>> 
BuildRangeExpression<TElement>(
    Expression<Func<TElement, long>> valueSelector, 
    IEnumerable<long> values)

そしてそれは次のように使用されます:

Expression<MyType, bool> match = 
    BuildRangeExpression<MyType, long>(my => my.ProductCode, productCodes);

var result = db.MyTypes.Where(match);

質問

どのように評価すればよいでしょうか

Expression<Func<TElement, long>> valueSelector

BuildRangeExpression現在ハードコードされている値の代わりに渡された値を使用できるように

long testValue = 5;
4

2 に答える 2

3

ブログ投稿のコードにはまさに必要なものがあると思います。必要なvalueSelector.Bodyのは、 your の代わりに使用Expression.Constant()し、生成された式に元のパラメーターを追加することだけです。

public static Expression<Func<TElement, bool>>
    BuildRangeExpression<TElement, TValue>(
    Expression<Func<TElement, TValue>> valueSelector,
    IEnumerable<Tuple<TValue, TValue>> values)
{
    var p = valueSelector.Parameters.Single();

    var equals = values.Select(
        tuple =>
        Expression.AndAlso(
            Expression.GreaterThanOrEqual(
                valueSelector.Body, Expression.Constant(tuple.Item1)),
            Expression.LessThanOrEqual(
                valueSelector.Body, Expression.Constant(tuple.Item2))));

    var body = equals.Aggregate(Expression.OrElse);

    return Expression.Lambda<Func<TElement, bool>>(body, p);
}
于 2013-02-18T22:26:23.910 に答える
1

を使用しExpression.Parameterます。

パラメータを作成します。

var param = Expression.Parameter(typeof(TElement), "arg")

の代わりにExpression.Constant(testvalue)、を置く必要がありますparam

次に、次のことを行う必要があります。

var result = Expression.Lambda<Func<TElement, bool>>(final, param).Compile() 
于 2013-02-18T22:22:38.077 に答える