我想构建一个使用另一个表达式的表达式,这会给我一个很好的ToString()
输出。
用普通的lambdas表示,我想这样做:
Func<string> extractFunc = () => "Marsh";
Func<bool> compareFunc = () => extractFunc() == "Mallow";
表达式相同:
Expression<Func<string>> extractExp = () => "Marsh";
Expression<Func<bool>> compareExp = () => extractExp.Compile()() == "Mallow";
执行compareExp.ToString()
会给我以下输出:
() => (Invoke(value(MyClass+<>c__DisplayClassb).extractExp.Compile()) == "Mallow")
我想要的是类似的东西:
() => "March" == "Mallow"
我应该写什么而不是extractExp.Compile()()
?
(我可能必须使用某种ExpressionVisitor
来获得漂亮的输出。如果答案包含了这个,那就是奖励,但更重要的是如何合并两个Expression
。)
答案 0 :(得分:3)
你在这里寻找的是Compose
方法,它可以采用表达式并将其映射到采用相同输入的方法,但是对第一个函数的结果执行操作:
public static Expression<Func<TResult>>
Compose<TIntermediate, TResult>(
this Expression<Func<TIntermediate>> first,
Expression<Func<TIntermediate, TResult>> second)
{
return Expression.Lambda<Func<TResult>>(
second.Body.Replace(second.Parameters[0], first.Body));
}
此方法使用以下方法将一个表达式的所有实例替换为另一个:
internal class ReplaceVisitor : ExpressionVisitor
{
private readonly Expression from, to;
public ReplaceVisitor(Expression from, Expression to)
{
this.from = from;
this.to = to;
}
public override Expression Visit(Expression node)
{
return node == from ? to : base.Visit(node);
}
}
public static Expression Replace(this Expression expression,
Expression searchEx, Expression replaceEx)
{
return new ReplaceVisitor(searchEx, replaceEx).Visit(expression);
}
现在你可以写:
Expression<Func<string>> extractExp = () => "Marsh";
Expression<Func<bool>> compareExp = extractExp.Compose(s => s == "Mallow");