合并表达式(Expression <Func <TIn,TOut >>与Expression <Func <TOut,bool >>)

时间:2019-07-17 12:16:36

标签: c# entity-framework linq lambda expression

我有两个表达式:

Expression<Func<T1, T2>> 
Expression<Func<T2, bool>> 

我想将它们结合起来,得到一个新的表达式,类型为Expression >,以便可以在Entity Framework LINQ中使用。

我在Expression.Invoke()的帮助下将它们组合在一起,但是它不起作用。

//Extensinon method
public static Expression<Func<T1, bool>> Compose<T1, T2>(this Expression<Func<T1, T2>> convertExpr, Expression<Func<T2, bool>> predicate) 
    => Expression.Lambda<Func<T1, bool>>(Expression.Invoke(predicate, convertExpr.Body), convertExpr.Parameters.First());
...
Expression<Func<One, Two>> convert;
Expression<Func<Two, bool>> predicate;
Expression<Func<One, bool>> filter=convert.Compose(predicate);

// Works fine
List<One> lst;
lst.AsQueryable().Where(filter);

DbSet<One> src;
// In next line throws runtime NotSupportedException: The LINQ expression node type 'Invoke' is not supported in LINQ to Entities
src.Where(filter);

更新

例如:

public class Book
{ 
  public int Id {get; protected set;}
  public string Name {get; protected set;}
  public virtual Genre {get; protected set;}
}

public class Genre
{
  public int Id {get; protected set;}
  public string Name {get; protected set;}
}
...
Expression<Func<Book, Genre>> convert = b=>b.Genre;
Expression<Func<Genre, bool>> predicate = g=>g.Name=="fantasy";
// Need: b=>b.Genre=="fantasy"
Expression<Func<Genre, bool>> filter=convert.Compose(predicate);

// Works fine
List<Book> lst;
lst.AsQueryable().Where(filter);

DbSet<Book> books;
// In next line throws runtime NotSupportedException: The LINQ expression node type 'Invoke' is not supported in LINQ to Entities
books.Where(filter);

2 个答案:

答案 0 :(得分:1)

使用通用的ExpressionVisitor将另一个Expression替换为另一个{我的标准Compose函数(我认为是更常见的数学顺序),替代了Body一个LambdaExpression代表另一个参数:

public static class ExpressionExt {
    // Compose: (y => f(y)).Compose(x => g(x)) -> x => f(g(x))
    /// <summary>
    /// Composes two LambdaExpression into a new LambdaExpression
    /// </summary>
    /// <param name="Tpg">Type of parameter to gFn, and type of parameter to result lambda.</param>
    /// <param name="Tpf">Type of result of gFn and type of parameter to fFn.</param>
    /// <param name="TRes">Type of result of fFn and type of result of result lambda.</param>
    /// <param name="fFn">The outer LambdaExpression.</param>
    /// <param name="gFn">The inner LambdaExpression.</param>
    /// <returns>LambdaExpression representing outer composed with inner</returns>
    public static Expression<Func<Tpg, TRes>> Compose<Tpg, Tpf, TRes>(this Expression<Func<Tpf, TRes>> fFn, Expression<Func<Tpg, Tpf>> gFn) =>
        Expression.Lambda<Func<Tpg, TRes>>(fFn.Body.Replace(fFn.Parameters[0], gFn.Body), gFn.Parameters[0]);

    /// <summary>
    /// Replaces an Expression (reference Equals) with another Expression
    /// </summary>
    /// <param name="orig">The original Expression.</param>
    /// <param name="from">The from Expression.</param>
    /// <param name="to">The to Expression.</param>
    /// <returns>Expression with all occurrences of from replaced with to</returns>
    public static Expression Replace(this Expression orig, Expression from, Expression to) => new ReplaceVisitor(from, to).Visit(orig);
}

/// <summary>
/// ExpressionVisitor to replace an Expression (that is Equals) with another Expression.
/// </summary>
public class ReplaceVisitor : ExpressionVisitor {
    readonly Expression from;
    readonly Expression to;

    public ReplaceVisitor(Expression from, Expression to) {
        this.from = from;
        this.to = to;
    }

    public override Expression Visit(Expression node) => node == from ? to : base.Visit(node);
}

有了这个功能,您的示例很简单:

Expression<Func<One, Two>> convert = p1 => new Two(p1);
Expression<Func<Two, bool>> predicate = p2 => p2 == new Two();
Expression<Func<One, bool>> filter = predicate.Compose(convert);

,尤其是对于EF表达式,在参数可能为{{的情况下,最好使用我的替代品来处理Invoke传播的null 1}}。它还使用与上述相同的null Replace

ExpressionVisitor

给出public static class ExpressionExt2 { public static Expression PropagateNull(this Expression orig) => new NullVisitor().Visit(orig); // Apply: (x => f).Apply(args) /// <summary> /// Substitutes an array of Expression args for the parameters of a lambda, returning a new Expression /// </summary> /// <param name="e">The original LambdaExpression to "call".</param> /// <param name="args">The Expression[] of values to substitute for the parameters of e.</param> /// <returns>Expression representing e.Body with args substituted in</returns> public static Expression Apply(this LambdaExpression e, params Expression[] args) { var b = e.Body; foreach (var pa in e.Parameters.Zip(args, (p, a) => (p, a))) b = b.Replace(pa.p, pa.a); return b.PropagateNull(); } } /// <summary> /// ExpressionVisitor to replace a null.member Expression with a null /// </summary> public class NullVisitor : System.Linq.Expressions.ExpressionVisitor { public override Expression Visit(Expression node) { if (node is MemberExpression nme && nme.Expression is ConstantExpression nce && nce.Value == null) return Expression.Constant(null, nce.Type.GetMember(nme.Member.Name).Single().GetMemberType()); else return base.Visit(node); } } public static class MeberInfoExt { public static Type GetMemberType(this MemberInfo member) { switch (member) { case FieldInfo mfi: return mfi.FieldType; case PropertyInfo mpi: return mpi.PropertyType; case EventInfo mei: return mei.EventHandlerType; default: throw new ArgumentException("MemberInfo must be if type FieldInfo, PropertyInfo or EventInfo", nameof(member)); } } } ,您的Apply很简单:

Compose

答案 1 :(得分:-1)

如果将它们组合在一起的唯一原因是要与Entity Framework一起使用,则可以使用Func而不是Expression

public static Func<T1, bool> Compose<T1, T2>(this Func<T1, T2> convert, Func<T2, bool> predicate)
{
   Func<T1, bool> combined = t1 => predicate(convert(t1));
   return combined;
}

以及用法

Func<One, Two> convert;
Func<Two, bool> predicate;
Func<One, bool> filter=convert.Compose(predicate);

DbSet<One> src;
src.Where(filter);

编辑

如果您确实需要使用表达式

 public static Expression<Func<T1, bool>> Combine<T1, T2>(this Expression<Func<T1, T2>> converterExpression,
    Expression<Func<T2, bool>> predicateExpression)
{
    var converter = converterExpression.Compile();
    var predicate = predicateExpression.Compile();

    Expression<Func<T1, bool>> exp = t1 => predicate(converter(t1));

    return exp;
}