これは、どのプロバイダーがそれを使用する必要があるかに大きく依存します。いくつかはうまくいくでしょう:
public static Expression<Func<TSource, TReturn>>
Merge<TSource, TSource2, TReturn>(
Expression<Func<TSource, TSource2>> foo1,
Expression<Func<TSource2, TReturn>> foo2)
{
return Expression.Lambda<Func<TSource, TReturn>>(
Expression.Invoke(foo2, foo1.Body),
foo1.Parameters);
}
ただし、他の人(EF)はそうしません。また、ビジターを使用して式ツリーを書き直し、式をインライン化することもできます。
public static Expression<Func<TSource, TReturn>>
Merge<TSource, TSource2, TReturn>(
Expression<Func<TSource, TSource2>> foo1,
Expression<Func<TSource2, TReturn>> foo2)
{
var swapped = new SwapVisitor(
foo2.Parameters.Single(), foo1.Body).Visit(foo2.Body);
return Expression.Lambda<Func<TSource, TReturn>>(
swapped, foo1.Parameters);
}
class SwapVisitor : ExpressionVisitor
{
private readonly Expression from, to;
public SwapVisitor(Expression from, Expression to)
{
this.from = from;
this.to = to;
}
public override Expression Visit(Expression node)
{
return node == from ? to : base.Visit(node);
}
}
これはすべてのプロバイダーで機能します。