5

あるクラスに直接適用される、または同じナビゲーションプロパティを持つ他のクラスに適用される同じ条件のセットがあります。

たとえば、私には家があり、エージェントがいます。エージェントは家に関連付けられています。

代理店名「a」の家を探している、または「a」という名前の代理店を探しているので、クエリは次のようになります。

 Expression<<Func<Agent,bool>> ax = x=> x.Name == "a" ;

 Expression<Func<Home,bool>> hx = x=> x.Agent.Name == "a";

エージェントの検索クエリは100近くありますが、それらをホームクエリ可能にも適用する必要があります。もう一度書くのは構わないが、開発の過程で頻繁に変わることがわかっているので、維持するのは難しい。

私がやりたいのは、このようなホームクエリの式を作成したいということです。

 Expression<Func<Home,bool>> hx = Combine( x=>x.Agent , x=>x.Name == "a");

コンバインが続くところ、

 Expression<Func<T,bool>> Combine<T,TNav>( 
     Expression<Func<T,TNav>> parent, 
     Expression<Func<TNav,bool>> nav){

     // combine above to form...


     (x=>x.Agent , x=>x.Name == "a") 
         => x => x.Agent.Name == "a"

     (x=>x.Agent, x=>x.Name.StartsWith("a") || x.Name.EndsWith("a"))
         => x=>x.Agent.Name.StartsWith("a") || x.Agent.Name.EndsWith("a")

     // look carefully, x gets replaced by x.Agent in every node..

     // I know very little about expression rewriting, so I need little help
 }
4

2 に答える 2

5

はい、元の表現の一部を置き換えるには訪問者が必要です。あなたはそのようなことをすることができます:

Expression<Func<T,bool>> Combine<T,TNav>(Expression<Func<T,TNav>> parent, Expression<Func<TNav,bool>> nav)
{
     var param = Expression.Parameter(typeof(T), "x");
     var visitor = new ReplacementVisitor(parent.Parameters[0], param);
     var newParentBody = visitor.Visit(parent.Body);
     visitor = new ReplacementVisitor(nav.Parameters[0], newParentBody);
     var body = visitor.Visit(nav.Body);
     return Expression.Lambda<Func<T, bool>>(body, param);
}

public class ReplacementVisitor : System.Linq.Expressions.ExpressionVisitor
{
    private readonly Expression _oldExpr;
    private readonly Expression _newExpr;
    public ReplacementVisitor(Expression oldExpr, Expression newExpr)
    {
        _oldExpr = oldExpr;
        _newExpr = newExpr;
    }

    public override Expression Visit(Expression node)
    {
        if (node == _oldExpr)
            return _newExpr;
        return base.Visit(node);
    }

}

使用例:

Expression<Func<Foo, Bar>> expr1 = f => f.Bar;
Expression<Func<Bar, bool>> expr2 = b => b.Baz;
var expr = Combine(expr1, expr2); // f => f.Bar.Baz
于 2012-06-05T14:27:00.903 に答える
0

PredicateBuilderの私のポイントを見てください、私はそれがあなたを助けると思います。そうでない場合は、必要なものをさらに詳しく説明してください。 Queryoverを使用した動的Where条件

于 2012-06-05T14:15:32.550 に答える