数据过滤
预定义的过滤
ISoftDelete
将实体标记为已删除,并不是物理删除. 实现 ISoftDelete
接口将你的实体"软删除".
示例:
namespace Acme.BookStore
{
public class Book : AggregateRoot<Guid>, ISoftDelete
{
public string Name { get; set; }
public bool IsDeleted { get; set; } //Defined by ISoftDelete
}
}
ISoftDelete
定义了 IsDeleted
属性. 当你使用仓储删除一条记录时, ABP 会自动将 IsDeleted
设置为 true,并将删除操作替换为修改操作(如果需要,也可以手动将 IsDeleted
设置为 true). 在查询数据库时会自动过滤软删除的实体.
ISoftDelete
过滤默认启用, 想要真正的从数据库删除实体需要显示的禁用过滤. 参见下面提到的IDataFilter
服务.
IMultiTenant
[多租户]是创建 SaaS 应用程序的有效方法. 多租户应用程序通常需要在租户间隔离数据. 实现 IMultiTenant
接口使你的实体支持 "多租户".
示例:
namespace Acme.BookStore
{
public class Book : AggregateRoot<Guid>, ISoftDelete, IMultiTenant
{
public string Name { get; set; }
public bool IsDeleted { get; set; } //Defined by ISoftDelete
public Guid? TenantId { get; set; } //Defined by IMultiTenant
}
}
IMultiTenant
接口定义了 TenantId
属性用于自动过滤当前租户实体. 更多信息参见[多租户]文档.
查询已删除的数据
你可以使用 IDataFilter
服务控制数据过滤.
示例:
namespace Acme.BookStore
{
public class MyBookService : ITransientDependency
{
private readonly IDataFilter _dataFilter;
private readonly IRepository<Book, Guid> _bookRepository;
public MyBookService(
IDataFilter dataFilter,
IRepository<Book, Guid> bookRepository)
{
_dataFilter = dataFilter;
_bookRepository = bookRepository;
}
public async Task<List<Book>> GetAllBooksIncludingDeletedAsync()
{
//Temporary disable the ISoftDelete filter
using (_dataFilter.Disable<ISoftDelete>())
{
return await _bookRepository.GetListAsync();
}
}
}
}
- [注入]
IDataFilter
服务到你的类中. - 在
using
语句中使用Disable
方法创建一个代码块,其中禁用了ISoftDelete
过滤器(始终与using
搭配使用,确保代码块执行后将过滤重置为之前的状态).
IDataFilter.Enable
方法可以启用过滤. 可以嵌套使用 Enable
和 Disable
方法定义内部作用域.
AbpDataFilterOptions
AbpDataFilterOptions
用于设置数据过滤系统.
下面的示例代码在默认情况下禁用了 ISoftDelete
过滤,除非显示启用,在查询数据库时会包含标记为已删除的实体:
Configure<AbpDataFilterOptions>(options =>
{
options.DefaultStates[typeof(ISoftDelete)] = new DataFilterState(isEnabled: false);
});
更改全局过滤的默认值需要小心,特别是在你使用预构建的模块时该模块可能是在默认启用软删除过滤的情况下开发的. 但你可以安全的为自己定义的数据过滤执行此操作.
自定义数据过滤
定义和实现新的过滤很大程序上取决与数据库提供者. ABP 为所有的数据库提供者实现了预构建的过滤.
首先为过滤定义一个接口 (如 ISoftDelete
和 IMultiTenant
) 然后用实体实现它.
示例:
public interface IIsActive
{
bool IsActive { get; }
}
IIsActive
接口可以过滤活跃/消极数据,任何[实体]都可以实现它:
public class Book : AggregateRoot<Guid>, IIsActive
{
public string Name { get; set; }
public bool IsActive { get; set; } //Defined by IIsActive
}
EntityFramework Core
ABP 使用EF Core 的全局过滤系统用于[EF Core 集成]. 所以它很好的集成到 EF Core 中,即使你直接使用 DbContext
它也可以正常工作.
实现自定义过滤的最佳方法是为重写你的 DbContext
的 ShouldFilterEntity
和 CreateFilterExpression
方法. 示例:
protected bool IsActiveFilterEnabled => DataFilter?.IsEnabled<IIsActive>() ?? false;
protected override bool ShouldFilterEntity<TEntity>(IMutableEntityType entityType)
{
if (typeof(IIsActive).IsAssignableFrom(typeof(TEntity)))
{
return true;
}
return base.ShouldFilterEntity<TEntity>(entityType);
}
protected override Expression<Func<TEntity, bool>> CreateFilterExpression<TEntity>()
{
var expression = base.CreateFilterExpression<TEntity>();
if (typeof(IIsActive).IsAssignableFrom(typeof(TEntity)))
{
Expression<Func<TEntity, bool>> isActiveFilter =
e => !IsActiveFilterEnabled || EF.Property<bool>(e, "IsActive");
expression = expression == null
? isActiveFilter
: CombineExpressions(expression, isActiveFilter);
}
return expression;
}
- 添加
IsActiveFilterEnabled
属性用于检查是否启用了IIsActive
. 内部使用了之前介绍到的IDataFilter
服务. - 重写
ShouldFilterEntity
和CreateFilterExpression
方法检查给定实体是否实现IIsActive
接口,在必要时组合表达式.