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));
});
另一种方法是使用原型表达式。
Expression<Func<object, bool>> filter =
e => EF.Property<bool>(e, "IsDeleted") == false
并使用参数替换器将参数与实际类型绑定:
.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 })
);
filter
是什么? - Hassan FaghihiExpression<Func<...>>(x=>x.IsDeleted)
的方法,或者以某种接近的方式。 - Hassan Faghihi