【问题标题】:Combine BinaryExpression and Expression<Func<dynamic, bool>> in C#在 C# 中结合 BinaryExpression 和 Expression<Func<dynamic, bool>>
【发布时间】:2018-07-24 11:08:27
【问题描述】:

如何组合BinaryExpressionExpression&lt;Func&lt;dynamic / T, bool&gt;&gt;

例如:

void AddGlobalFilter<T>(Expression<Func<T, bool>> expr)
{
    var parameter = Expression.Parameter(type, "t");
    var member = Expression.Property(filter.Parameter, field);
    var constant = Expression.Constant(null);
    var body = Expression.Equal(member, constant);

    var combine = Expression.AndAlso(body, expr);
}

我正在尝试为实体框架 (EF) 核心定义 global filter。问题是我必须manually combine multiple filters

如果模型实现了IDbDeleted接口,则可以在ModelBuilder中添加一个过滤器。
可以为特定型号手动添加另一个。基本思想是我有一个所有表达式的列表,然后将它们组合起来:

var expression = listExpressions.First();
foreach (var second in listExpressions.Skip(1))
{
    expression = Expression.AndAlso(expression, second);
}
var lambdaExpression = Expression.Lambda(expression, parameter);
modelBuilder.Entity(item.Key).HasQueryFilter(lambdaExpression);

当然会出错(第一个来自Expression.Equal,第二个来自t =&gt; t...):

过滤表达式 't => t => (Not(t. ...

已编辑:代码看起来像这样:

[Table("MyEntities")]
public class DbMyEntity : IDeleted
{
    public string Name { get; set; }
    public DateTime? DateTimeDeleted { get; set; }
}

public interface IDeleted
{
    DateTime? DateTimeDeleted { get; set; }
}

public class MyContext : IdentityDbContext
{
    private Dictionary<Type, List<Expression>> dict = new Dictionary<Type, List<Expression>>();
    private Dictionary<Type, ParameterExpression> dictParameter = new Dictionary<Type, ParameterExpression>();

    private ParameterExpression GetParameter(Type type)
    {
        if (!this.dictParameter.ContainsKey(type))
        {
            this.dictParameter.Add(type, Expression.Parameter(type, "t"));
        }
        return this.dictParameter[type];
    }

    private void AddToDict(Type type, Expression expr)
    {
        if (!this.dict.ContainsKey(type))
        {
            this.dict.Add(type, new List<Expression>());
            this.GetParameter(type);  //Just to create ParameterExpression if not exists.
        }

        this.dict[type].Add(expr);
    }

    private void AddToDict<T>(Expression<Func<T, bool>> expr)
    {
        this.AddToDict(typeof(T), expr);
    }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {

        base.OnModelCreating(modelBuilder);

        foreach (var entity in modelBuilder.Model.GetEntityTypes())
        {
            if (typeof(IDeleted).IsAssignableFrom(entity.ClrType))
            {
                var member = Expression.Property(this.GetParameter(entity.ClrType), "DateTimeDeleted");
                var constant = Expression.Constant(null);
                var body = Expression.Equal(member, constant);
                this.AddToDict(entity.ClrType, body);
            }
        }

        //This is done in another project in same solution. See comment bellow.
        this.AddToDict<DbMyEntity>(t => t.Name == null || t.Name == "Something");
        //foreach (var builderType in allDllModules)
        //{
        //    if (builderType != null && builderType != typeof(ICustomModelBuilder))
        //    {
        //        var builder = (ICustomModelBuilder)Activator.CreateInstance(builderType);
        //        builder.Build(modelBuilder);
        //    }
        //}

        foreach (var item in this.dict)
        {
            var expression = item.Value.First();
            foreach (var second in item.Value.Skip(1))
            {
                expression = Expression.AndAlso(expression, second);
            }
            var lambdaExpression = Expression.Lambda(expression, this.dictParameter[item.Key]);
            modelBuilder.Entity(item.Key).HasQueryFilter(lambdaExpression);
        }
    }
}

【问题讨论】:

  • 您能分享一下您当前代码的minimal reproducible example 吗?您的“完整”代码远不能编译
  • @CamiloTerevinto 已修复。
  • 我正要回答,然后我看到了this。将ReplaceExpressionVisitor 与一个小改动结合起来:不要将body 添加到字典中,而是添加var asLambda = Expression.Lambda(body, parameter);,使其最终成为Expression&lt;Func&lt;T, bool&gt;&gt;

标签: c# lambda entity-framework-core expression-trees entity-framework-core-2.1


【解决方案1】:

您将expressionslambda expressions 混合在一起。有很多帖子展示了如何组合 lambda 表达式,但最重要的部分是从 lambda 表达式 bodies 组合表达式并重新绑定 parameters

后者通常通过自定义ExpressionVisitor 来实现,如下所示:

using System.Linq.Expressions;

public static class ExpressionExtensions
{
    public static Expression ReplaceParameter(this Expression expression, ParameterExpression source, Expression target)
    {
        return new ParameterReplacer { Source = source, Target = target }.Visit(expression);
    }

    class ParameterReplacer : ExpressionVisitor
    {
        public ParameterExpression Source;
        public Expression Target;
        protected override Expression VisitParameter(ParameterExpression node)
        {
            return node == Source ? Target : base.VisitParameter(node);
        }
    }
}

现在关于 EF Core 组合查询过滤器。

使用字典和表达式列表对于您正在做的事情来说似乎过于复杂了。由于IMutableEntityType 提供对QueryFilter 的读/写访问,因此可以使用一小组自定义扩展方法来实现。

他们都像这样进入一个类:

using System;
using System.Linq.Expressions;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Metadata.Builders;

public static class QueryFilterExtensions
{
}

第一种方法:

public static void AddQueryFilter(this IMutableEntityType target, LambdaExpression filter)
{
    if (target.QueryFilter == null)
        target.QueryFilter = filter;
    else
    {
        var parameter = target.QueryFilter.Parameters[0];
        var left = target.QueryFilter.Body;
        var right = filter.Body.ReplaceParameter(filter.Parameters[0], parameter);
        var body = Expression.AndAlso(left, right);
        target.QueryFilter = Expression.Lambda(body, parameter);
    }
}

这是一种非泛型方法,它使用AndAlso (C# &amp;&amp;) 运算符将现有过滤器与通过过滤器组合起来,并展示了上述 lambda 表达式组合原理。

但是,它并没有那么直接有用,例如在您的实体类型配置循环中(它可以,但需要您手动构建 lambda 表达式,而不是让 C# 编译器这样做)。于是就出现了第二种方法:

public static void AddQueryFilter<T>(this IMutableEntityType target, Expression<Func<T, bool>> filter)
{
    LambdaExpression targetFilter = filter;
    if (target.ClrType != typeof(T))
    {
        var parameter = Expression.Parameter(target.ClrType, "e");
        var body = filter.Body.ReplaceParameter(filter.Parameters[0], parameter);
        targetFilter = Expression.Lambda(body, parameter);
    }
    target.AddQueryFilter(targetFilter);
}

这是一个通用方法 - 不是完全类型安全的,但允许您使用编译时 lambda 表达式并将其绑定到实际的实体类型,如下所示:

foreach (var entityType in modelBuilder.Model.GetEntityTypes())
{
    if (typeof(IDeleted).IsAssignableFrom(entityType.ClrType))
        entityType.AddQueryFilter<IDeleted>(e => e.DateTimeDeleted == null);
}

看起来更好,不是吗:)

最后一个自定义扩展方法是对标准 EF Core 通用 HasQueryFilter 方法的补充(替换):

public static EntityTypeBuilder<TEntity> AddQueryFilter<TEntity>(this EntityTypeBuilder<TEntity> target, Expression<Func<TEntity, bool>> filter)
    where TEntity : class
{
    target.Metadata.AddQueryFilter(filter);
    return target;
}

并允许您替换

this.AddToDict<DbMyEntity>(t => t.Name == null || t.Name == "Something");

用起来更方便

modelBuilder.Entity<DbMyEntity>()
    .AddQueryFilter(t => t.Name == null || t.Name == "Something");

更新(EF Core 3.0): QueryFilter 属性已替换为 GetQueryFilterSetQueryFilter 扩展方法。

【讨论】:

  • 这看起来很棒。请给我一两天时间来深入研究代码并尝试一下。
  • 抱歉拖了这么久。太完美了!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-11-26
  • 1970-01-01
  • 2013-03-09
  • 1970-01-01
相关资源
最近更新 更多