构建执行另一个表达式的表达式

时间:2014-07-01 20:58:13

标签: c# linq-expressions

我想构建一个使用另一个表达式的表达式,这会给我一个很好的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。)

1 个答案:

答案 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");