gpt4 book ai didi

c# - 组合两个 linq 表达式以注入(inject)导航属性

转载 作者:太空狗 更新时间:2023-10-30 01:23:54 24 4
gpt4 key购买 nike

我有一组相同的条件直接应用于一个类或具有相同导航属性的其他一些类。

比如我有家,我有代理,代理与家相关。

所以我正在寻找代理名称为“a”的房屋,或者我正在寻找名称为“a”的代理,查询如下,

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

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

我有将近 100 个关于 Agent 的搜索查询,我也必须将它们应用到 Home queryable。我不介意重新写一遍,但是很难维护,因为我们知道它们在开发过程中会经常更改。

我想做的是,我想像这样为 Home 查询编写表达式,

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

Combine 将跟随的地方,

 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
}

最佳答案

是的,您确实需要访问者来替换部分原始表达式。你可以这样做:

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

关于c# - 组合两个 linq 表达式以注入(inject)导航属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10898800/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com