可以不看下面內容,直接看這:https://www.cnblogs.com/MindSharing/p/11283914.html
================================
ABP官方數據過濾的地址:https://aspnetboilerplate.com/Pages/Documents/Data-Filters
中文可以看這個:https://aspnetboilerplate.com/Pages/Documents/Data-Filters
看完后最大問題就是,自定義的數據過濾和UOW中事務是在一起的
using (CurrentUnitOfWork.EnableFilter("PersonFilter"))
{
using(CurrentUnitOfWork.SetFilterParameter("PersonFilter", "personId", 42))
{
var phones = _phoneRepository.GetAllList();
//...
}
}
總不可能每次我寫服務的時候,都在方法里面寫一串這個代碼吧,那這樣也沒方便多少
所以可以考慮使用我方法來給過濾器全局設置一個值。
我的應用場景是這樣的,我的項目登錄的時候會讓用戶選擇一個專業,所有數據都是這個專業下的,什么建築啊,結構啊

為了實現這個篩選,先還是需要建立過濾器,在Core項目添加一個接口
public interface IHasMajor
{
string Major { get; set; }
}
在EF項目中的DbContext.cs 中注冊 過濾器
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Filter("MajorFilter", (IHasMajor entity, string major) => entity.Major == major, "");
}
在你需要篩選專業的實體里面實現這個接口
[Table("DictionaryInfo")]
public class DictionaryInfo : FullAuditedEntity, IHasMajor
{
public virtual int? ParentId { get; set; }
public virtual string Value { get; set; }
public virtual string Description { get; set; }
public virtual DictionaryInfo ParentDictionaryInfo { get; set; }
public virtual ICollection<DictionaryInfo> ChildDictionaryInfos { get; set; }
public virtual string Major { get; set; }
}
好了下面就是關鍵的了,傳統做法,我們就是在服務里面設置數據過濾的參數
public async Task<GetDictionaryInfoOutput> GetAllDictionary()
{
using (CurrentUnitOfWork.SetFilterParameter(SysConsts.DataFilter.MajorFilter, SysConsts.DataFilter.MajorParameter, AbpSession.GetMajor()))
{
List<DictionaryInfo> dictionaries = await _dictionaryInfoRepository.GetAllListAsync();
var result = new GetDictionaryInfoOutput
{
Dictionary = dictionaries.MapTo<List<DictionaryInfoDto>>()
};
return result;
}
}
這種方式感覺很不優雅,改用攔截器來自動為每個服務注入 過濾器參數,
ABP 官方對攔截器他也給了個例子,可以看這個
https://www.codeproject.com/Articles/1080517/Aspect-Oriented-Programming-using-Interceptors-wit
添加個一個攔截器,在應用層
public class FilterInterceptor : IInterceptor
{
private readonly IAbpSession _abpSession;
private readonly IUnitOfWork _unitOfWork;
private readonly IUnitOfWorkManager _unitOfWorkManager;
public FilterInterceptor(
IUnitOfWork unitOfWork
, IAbpSession abpSession
, IUnitOfWorkManager unitOfWorkManager)
{
Logger = NullLogger.Instance;
_unitOfWork = unitOfWork;
_abpSession = abpSession;
_unitOfWorkManager = unitOfWorkManager;
}
public ILogger Logger { get; set; }
public void Intercept(IInvocation invocation)
{
string major = _abpSession.GetMajor();
_unitOfWorkManager.Current.SetFilterParameter(SysConsts.DataFilter.MajorFilter, SysConsts.DataFilter.MajorParameter, major);
//Executing the actual method
invocation.Proceed();
}
}
這里面有一個主意地方,攔截器請注入IUnitOfWorkManager ,而不是注入IUnitOfWork ,后者設置參數了,沒效果
然后再注冊攔截器,在應用層添加一個文件
public static class InterceptorRegistrar
{
public static void Initialize(IKernel kernel)
{
kernel.ComponentRegistered += Kernel_ComponentRegistered;
}
private static void Kernel_ComponentRegistered(string key, IHandler handler)
{
if (typeof(IApplicationService).IsAssignableFrom(handler.ComponentModel.Implementation))
{
handler.ComponentModel.Interceptors.Add(new InterceptorReference(typeof(FilterInterceptor)));
}
}
}
在應用層的ApplicationModule.cs 文件里面的PreInitialize方法里面
public override void PreInitialize()
{
InterceptorRegistrar.Initialize(IocManager.IocContainer.Kernel);
}
好這樣就注冊完了,可以試試服務層每個方法都能自動判斷專業了,把服務里面手動設置篩選參數的地方都可以去掉了
public async Task<GetDictionaryInfoOutput> GetAllDictionary()
{
//using (CurrentUnitOfWork.SetFilterParameter(SysConsts.DataFilter.MajorFilter, SysConsts.DataFilter.MajorParameter, AbpSession.GetMajor()))
//{
List<DictionaryInfo> dictionaries = await _dictionaryInfoRepository.GetAllListAsync();
var result = new GetDictionaryInfoOutput
{
Dictionary = dictionaries.MapTo<List<DictionaryInfoDto>>()
};
return result;
//}
}
專業的值是存在session里面的,如何把自己想要的值存到session 里面,可以參照這篇文章
http://www.jianshu.com/p/930c10287e2a
