3

考虑以下设置:

class A { public int x; }
class B { public int y; }

static class Helper
{
    public static Expression<Func<B>> BindInput(
        Expression<Func<A, B>> expression,
        A input)
    {
        //TODO
    }
}

static void Main(string[] args)
{
    Expression<Func<B>> e = Helper.BindInput(
        (A a) => new B { y = a.x + 3 },
        new A { x = 4 });

    Func<B> f = e.Compile();
    Debug.Assert(f().y == 7);
}

我想要在该方法中做的BindInput是将表达式转换为input嵌入其中。在 中的示例用法中Main,结果表达式e将是

() => new B { y = input.x + 3 }

input传入的第二个值在哪里BindInput

我该怎么做呢?

编辑:

我应该补充一点,以下表达式e不是想要的:

((A a) => new B { y = a.x + 3 })(input)

这将是相当微不足道的,因为它只涉及在现有表达式的顶部添加一个层。

4

1 回答 1

4

经过大量搜索,我偶然发现了魔法ExpressionVisitor课。以下似乎工作完美:

class MyExpressionVisitor : ExpressionVisitor
{
    public ParameterExpression TargetParameterExpression { get; private set; }
    public object TargetParameterValue { get; private set; }
    public MyExpressionVisitor(ParameterExpression targetParameterExpression, object targetParameterValue)
    {
        this.TargetParameterExpression = targetParameterExpression;
        this.TargetParameterValue = targetParameterValue;
    }
    protected override Expression VisitParameter(ParameterExpression node)
    {
        if (node == TargetParameterExpression)
            return Expression.Constant(TargetParameterValue);
        return base.VisitParameter(node);
    }
}

static class Helper
{
    public static Expression<Func<B>> BindInput(Expression<Func<A, B>> expression, A input)
    {
        var parameter = expression.Parameters.Single();
        var visitor = new MyExpressionVisitor(parameter, input);
        return Expression.Lambda<Func<B>>(visitor.Visit(expression.Body));
    }
}
于 2013-01-10T00:31:48.110 回答