另一种解决方案是使用ExpressionVisitor将右侧表达式中的参数替换为整个左侧表达式,也就是说,将左侧嵌入到右侧。
表达式访问者将非常简单,将所需的数据添加到构造函数中,重写一个方法就可以了。
internal sealed class ParameterReplaceVisitor : ExpressionVisitor
{
private readonly ParameterExpression _searched;
private readonly Expression _replaced;
public ParameterReplaceVisitor(ParameterExpression searched, Expression replaced)
{
if (searched == null)
throw new ArgumentNullException(nameof(searched));
if (replaced == null)
throw new ArgumentNullException(nameof(replaced));
_searched = searched;
_replaced = replaced;
}
protected override Expression VisitParameter(ParameterExpression node)
{
if (node == _searched)
return _replaced;
return base.VisitParameter(node);
}
}
它可以很容易地扩展来处理构造函数中的表达式集合,但我保持简短。
现在,您只需在表达式主体上使用它并构造新的 lambda。
private static Expression<Func<TIn, TOut>> Merge<TIn, TInter, TOut>(Expression<Func<TIn, TInter>> left, Expression<Func<TInter, TOut>> right)
{
var merged = new ParameterReplaceVisitor(right.Parameters[0], left.Body).Visit(right.Body);
var lambda = Expression.Lambda<Func<TIn, TOut>>(merged, left.Parameters[0]);
return lambda;
}
我在这段代码上测试过:
Expression<Func<string, int>> l = s => s.Length + 5;
Expression<Func<int, string>> r = i => i.ToString() + " something";
var merged = Merge(l, r);
var res = merged.Compile()("test");
结果如预期:9 something。
编辑:
如果您关心性能,为什么要使用表达式而不是简单的Funcs?然后你可以一个接一个地调用。后面会分析表达式树吗?