gpt4 book ai didi

c# - EF 核心 : Soft delete with shadow properties and query filters

转载 作者:太空狗 更新时间:2023-10-29 17:31:30 25 4
gpt4 key购买 nike

我创建了一个界面来尝试执行软删除,混合影子属性和查询过滤器。但它不起作用。

public interface IDeletableEntity {}

然后在我的模型构建器中

 builder.Model.GetEntityTypes()
.Where(entityType => typeof(IDeletableEntity).IsAssignableFrom(entityType.ClrType))
.ToList()
.ForEach(entityType =>
{
builder.Entity(entityType.ClrType).Property<Boolean>("IsDeleted");
builder.Entity(entityType.ClrType).HasQueryFilter(e => EF.Property<Boolean>(e, "IsDeleted") == false);
});

但是带有查询过滤器的行没有编译。我得到的错误是“无法将 lambda 表达式转换为类型‘lambda 表达式’,因为它不是委托(delegate)类型”

如果我这样做,它就会起作用。

builder.Entity<MyEntity>().HasQueryFilter(m => EF.Property<Boolean>(m, "IsDeleted") == false);

有什么办法可以做到这一点?这是为了与 IDeletableEntity 有一个接口(interface),而不是必须这样做,在我想使用软删除实体的每个实体中

非常感谢,

最佳答案

HasQueryFilter非通用的 EntityTypeBuilder (与通用的 EntityTypeBuilder<TEntity> 相反)几乎无法使用,因为没有简单的方法来创建预期的 LambdaExpression .

一种解决方案是使用 Expression 手动构建 lambda 表达式类方法:

.ForEach(entityType =>
{
builder.Entity(entityType.ClrType).Property<Boolean>("IsDeleted");
var parameter = Expression.Parameter(entityType.ClrType, "e");
var body = Expression.Equal(
Expression.Call(typeof(EF), nameof(EF.Property), new[] { typeof(bool) }, parameter, Expression.Constant("IsDeleted")),
Expression.Constant(false));
builder.Entity(entityType.ClrType).HasQueryFilter(Expression.Lambda(body, parameter));
});

另一种是使用原型(prototype)表达式

Expression<Func<object, bool>> filter = 
e => EF.Property<bool>(e, "IsDeleted") == false;

并使用参数替换器将参数与实际类型绑定(bind):

.ForEach(entityType =>
{
builder.Entity(entityType.ClrType).Property<Boolean>("IsDeleted");
var parameter = Expression.Parameter(entityType.ClrType, "e");
var body = filter.Body.ReplaceParameter(filter.Parameters[0], parameter);
builder.Entity(entityType.ClrType).HasQueryFilter(Expression.Lambda(body, parameter));
});

哪里ReplaceParameter是我用于表达式树操作的自定义辅助扩展方法之一:

public static partial class ExpressionUtils
{
public static Expression ReplaceParameter(this Expression expr, ParameterExpression source, Expression target) =>
new ParameterReplacer { Source = source, Target = target }.Visit(expr);

class ParameterReplacer : System.Linq.Expressions.ExpressionVisitor
{
public ParameterExpression Source;
public Expression Target;
protected override Expression VisitParameter(ParameterExpression node) => node == Source ? Target : node;
}
}

但在我看来,最自然的解决方案是将配置代码移动到通用方法中并通过反射调用它。例如:

static void ConfigureSoftDelete<T>(ModelBuilder builder)
where T : class, IDeletableEntity
{
builder.Entity<T>().Property<Boolean>("IsDeleted");
builder.Entity<T>().HasQueryFilter(e => EF.Property<bool>(e, "IsDeleted") == false);
}

然后

.ForEach(entityType => GetType()
.GetMethod(nameof(ConfigureSoftDelete), BindingFlags.NonPublic | BindingFlags.Static)
.MakeGenericMethod(entityType.ClrType)
.Invoke(null, new object[] { builder })
);

关于c# - EF 核心 : Soft delete with shadow properties and query filters,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47673524/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com