Filter 是延續 ASP.NET MVC 的產物,同樣保留了五種的 Filter,分別是 Authorization Filter 、Resource Filter 、Action Filter 、Exception Filter  及 Result Filter 。
iT 邦幫忙 2018 鐵人賽 - Modern Web 組參賽文章:[Day14] ASP.NET Core 2 系列 - Filters 
Filter 介紹 Filter 的作用是在 Action 執行前 或執行後 做一些加工處理。
ASP.NET Core 有以下五種 Filter 可以使用:
Authorization Filter Resource Filter Action Filter Exception Filter Result Filter Filter 運作方式 ASP.NET Core 的每個 Request 都會先經過已註冊的 Middleware 接著才會執行 Filter,除了會依照上述的順序外,同類型的 Filter 預設都會以先進後出的方式處裡封包。
建立 Filter ASP.NET Core 的 Filter 基本上跟 ASP.NET MVC 的差不多。
Authorization Filter AuthorizationFilter.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 using  Microsoft.AspNetCore.Http;using  Microsoft.AspNetCore.Mvc.Filters;namespace  MyWebsite.Filters {     public  class  AuthorizationFilter  : IAuthorizationFilter      {         public  void  OnAuthorization (AuthorizationFilterContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         }     } } 
非同步的方式:
1 2 3 4 5 6 7 8 public  class  AuthorizationFilter  : IAsyncAuthorizationFilter {     public  async  Task OnAuthorizationAsync (AuthorizationFilterContext context )     {         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );     } } 
Resource Filter ResourceFilter.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 using  Microsoft.AspNetCore.Http;using  Microsoft.AspNetCore.Mvc.Filters;namespace  MyWebsite.Filters {     public  class  ResourceFilter  : IResourceFilter      {         public  void  OnResourceExecuting (ResourceExecutingContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         }         public  void  OnResourceExecuted (ResourceExecutedContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );         }     } } 
非同步的方式:
1 2 3 4 5 6 7 8 9 10 11 12 public  class  ResourceFilter  : IAsyncResourceFilter {     public  async  Task OnResourceExecutionAsync (ResourceExecutingContext context, ResourceExecutionDelegate next )     {         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         await  next();         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );     } } 
Action Filter ActionFilter.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 using  Microsoft.AspNetCore.Http;using  Microsoft.AspNetCore.Mvc.Filters;namespace  MyWebsite.Filters {     public  class  ActionFilter  : IActionFilter      {         public  void  OnActionExecuting (ActionExecutingContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         }         public  void  OnActionExecuted (ActionExecutedContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );         }     } } 
非同步的方式:
1 2 3 4 5 6 7 8 9 10 11 12 public  class  ActionFilter  : IAsyncActionFilter {     public  async  Task OnActionExecutionAsync (ActionExecutingContext context, ActionExecutionDelegate next )     {         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         await  next();         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );     } } 
Result Filter ResultFilter.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 using  Microsoft.AspNetCore.Http;using  Microsoft.AspNetCore.Mvc.Filters;namespace  MyWebsite.Filters {     public  class  ResultFilter  : IResultFilter      {         public  void  OnResultExecuting (ResultExecutingContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         }         public  void  OnResultExecuted (ResultExecutedContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );         }     } } 
非同步的方式:
1 2 3 4 5 6 7 8 9 10 11 12 public  class  ResultFilter  : IAsyncResultFilter {     public  async  Task OnResultExecutionAsync (ResultExecutingContext context, ResultExecutionDelegate next )     {         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         await  next();         await  context.HttpContext.Response.WriteAsync($"{GetType().Name}  out. \r\n" );     } } 
Exception Filter ExceptionFilter.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 using  Microsoft.AspNetCore.Http;using  Microsoft.AspNetCore.Mvc.Filters;namespace  MyWebsite.Filters {     public  class  ExceptionFilter  : IExceptionFilter      {         public  void  OnException (ExceptionContext context )         {             context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         }     } } 
非同步的方式:
1 2 3 4 5 6 7 8 9 public  class  ExceptionFilter  : IAsyncExceptionFilter {     public  Task OnExceptionAsync (ExceptionContext context )     {         context.HttpContext.Response.WriteAsync($"{GetType().Name}  in. \r\n" );         return  Task.CompletedTask;     } } 
註冊 Filter Filter 有兩種註冊方式,一種是全域註冊,另一種是用 [Attribute] 區域註冊的方式,只套用在特定的 Controller 或 Action。
全域註冊 在 Startup.ConfigureServices 的 MVC 服務中註冊 Filter,這樣就可以套用到所有的 Request。如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 public  class  Startup {     public  void  ConfigureServices (IServiceCollection services )     {         services.AddMvc(config =>         {             config.Filters.Add(new  ResultFilter());             config.Filters.Add(new  ExceptionFilter());             config.Filters.Add(new  ResourceFilter());         });     } } 
區域註冊 ASP.NET Core 在區域註冊 Filter 的方式跟 ASP.NET MVC 有一點不一樣,要透過 [TypeFilter(type)]。[TypeFilter(type)] 就可以區域註冊 Filter。如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 namespace  MyWebsite.Controllers {     [TypeFilter(typeof(AuthorizationFilter)) ]     public  class  HomeController  : Controller      {         [TypeFilter(typeof(ActionFilter)) ]         public  void  Index ()         {             Response.WriteAsync("Hello World! \r\n" );         }                  [TypeFilter(typeof(ActionFilter)) ]         public  void  Error ()         {             throw  new  System.Exception("Error" );         }     } } 
[TypeFilter(type)] 用起來有點冗長,想要像過去 ASP.NET MVC 用 [Attribute] 註冊 Filter 的話,只要將 Filter 繼承 Attribute 即可。如下:
1 2 3 4 5 6 7 8 public  class  AuthorizationFilter  : Attribute , IAuthorizationFilter {      } public  class  ActionFilter  : Attribute , IActionFilter {      } 
[Attribute] 註冊就可以改成如下方式:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 namespace  MyWebsite.Controllers {     [AuthorizationFilter ]     public  class  HomeController  : Controller      {         [ActionFilter ]         public  void  Index ()         {             Response.WriteAsync("Hello World! \r\n" );         }                  [ActionFilter ]         public  void  Error ()         {             throw  new  System.Exception("Error" );         }     } } 
執行結果 http://localhost:5000/Home/Index 輸出結果如下:
1 2 3 4 5 6 7 8 AuthorizationFilter in. ResourceFilter in. ActionFilter in. Hello World! ActionFilter out. ResultFilter in. ResultFilter out. ResourceFilter out. 
http://localhost:5000/Home/Error 輸出結果如下:
1 2 3 4 5 6 AuthorizationFilter in. ResourceFilter in. ActionFilter in. ActionFilter out. ExceptionFilter in. ResourceFilter out. 
執行順序 預設註冊同類型的 Filter 是以先進後出的方式處裡封包,註冊層級也會影響執行順序。
但也可以透過實作 IOrderedFilter 更改執行順序。例如:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 public  class  ActionFilter  : Attribute , IActionFilter , IOrderedFilter {     public  string  Name { get ; set ; }     public  int  Order { get ; set ; } = 0 ;     public  void  OnActionExecuting (ActionExecutingContext context )     {         context.HttpContext.Response.WriteAsync($"{GetType().Name} ({Name} ) in. \r\n" );     }     public  void  OnActionExecuted (ActionExecutedContext context )     {         context.HttpContext.Response.WriteAsync($"{GetType().Name} ({Name} ) out. \r\n" );     } } 
在註冊 Filter 時帶上 Order,數值越小優先權越高。
1 2 3 4 5 6 7 8 9 10 11 public  class  Startup {     public  void  ConfigureServices (IServiceCollection services )     {         services.AddMvc(config =>         {             config.Filters.Add(new  ActionFilter() { Name = "Global" , Order = 3  });         });     } } 
1 2 3 4 5 6 7 8 9 10 11 12 13 namespace  MyWebsite.Controllers {     [ActionFilter(Name = "Controller" , Order = 2) ]     public  class  HomeController  : Controller      {         [ActionFilter(Name = "Action" , Order = 1) ]         public  void  Index ()         {             Response.WriteAsync("Hello World! \r\n" );         }     } } 
變更執行順序後的輸出內容:
1 2 3 4 5 6 7 ActionFilter(Action) in.  ActionFilter(Controller) in.  ActionFilter(Global) in.  Hello World!  ActionFilter(Global) out.  ActionFilter(Controller) out.  ActionFilter(Action) out.  
參考 ASP.NET Core Filters