ASP.NET Core-ActionFilter實現依賴注入(ServiceFilterAttribute 、TypeFilterAttribute) 【轉】


一、簡介

  前幾篇文章都是講ASP.NET Core MVC中的依賴注入(DI)與擴展點的,也許大家都發現在ASP.NET CORE中所有的組件都是通過依賴注入來擴展的,而且面向一組功能就會有一組接口或抽象工廠來擴展功能,就如IControllerActivator這樣的功能點在上篇文章(查看.NET Core源代碼通過Autofac實現依賴注入到Controller屬性)中也提到了,今天我們主要介紹一個大類似的擴展點,ASP.NET Core MVC中為我們提供了新的機制為Action Filters(也就是過濾器)進行依賴注入的擴展。

二、過濾器依賴注入

  在ASP.NET Core MVC中,框架中為我們提供了類型為 IFilter 的 Attributes 來裝飾Action,用於攔截Action請求,這有在以前的版本中就有了,但是如果我們想結合依賴注入使用的話要使用IFilterFactory接口來擴展Action Filter的創建過程。

  2.1 IFilterFactory接口定義

public interface IFilterFactory : IFilter
{
    IFilter CreateInstance([NotNull] IServiceProvider serviceProvider);
}

 

  我們想要創建一個Filter Attribute並需要依賴注入的話一般想要的代碼為:

復制代碼
public class FilterClass : ActionFilterAttribute  
{
  public FilterClass(IDependency1 dependency1, IDependency2 dependency2)
  {
    // ...use dependencies
  }
}
復制代碼

 

  ASP.NET Core MVC中為我們提供了兩種簡單的IFilterFactory ServiceFilterAttribute 和 TypeFilterAttribute 。來個例子看看怎么使用。

復制代碼
public class HomeController: Controller  
{
    [TypeFilter(typeof(FilterClass))]
    [ServiceFilter(typeof(FilterClass))]
    public IActionResult Index()
    {
        return View();
    }
}
復制代碼

  2.2 ServiceFilterAttribute 

   其實看到名字,有些朋友就能想到了,它是基於依賴注入的一個IFilterFactory,Service這個詞強化了它是一個通過獲取服務來實現依賴注入的,大家想到了什么?是不是GetService()? 沒錯,其實它的機制就是這個。

     要使用ServiceFilter就必須在依賴注入容器里注冊對應的類型,比如下面的例子就要先將FilterClass類型注冊到IOC容器里。

public void ConfigureServices(IServiceCollection services)
{
      services.AddSingleton<FilterClass>();
 
      services.AddMvc()
}

  當然如果FilterClass類型的構造器需要注入類型時,也需要在IOC容器里進行注冊才可以使用。

  我們來看下ServiceFilterAttribute的源代碼:

復制代碼
public class ServiceFilterAttribute : Attribute, IFilterFactory, IOrderedFilter
{
    public ServiceFilterAttribute([NotNull] Type type)
    {
        ServiceType = type;
    }

    public Type ServiceType { get; private set; }

    public int Order { get; set; }

    public IFilter CreateInstance([NotNull] IServiceProvider serviceProvider)
    {
        var service = serviceProvider.GetRequiredService(ServiceType);

        var filter = service as IFilter;
        if (filter == null)
        {
            throw new InvalidOperationException(Resources.FormatFilterFactoryAttribute_TypeMustImplementIFilter(
                typeof(ServiceFilterAttribute).Name,
                typeof(IFilter).Name));
        }

        return filter;
    }
}
復制代碼

  2.3 TypeFilterAttribute

   當然你也可以選擇使用這個類似於ServiceFilter過濾器的TypeFilter過濾器,它也同樣實現了IFilterFactory接口,並可以通過它創建出可使用依賴注入的過濾器來。之所以叫TypeFilter就是因為它並不需要在依賴注入容器里注冊類型就能創建出過濾器,  我們來看下它的代碼:

復制代碼
public class TypeFilterAttribute : Attribute, IFilterFactory, IOrderedFilter
{
    private ObjectFactory factory;

    public TypeFilterAttribute([NotNull] Type type)
    {
        ImplementationType = type;
    }

    public object[] Arguments { get; set; }

    public Type ImplementationType { get; private set; }

    public int Order { get; set; }

    public IFilter CreateInstance([NotNull] IServiceProvider serviceProvider)
    {
        if (this.factory == null)
        {
            var argumentTypes = Arguments?.Select(a => a.GetType())?.ToArray();

            this.factory = ActivatorUtilities.CreateFactory(ImplementationType, argumentTypes ?? Type.EmptyTypes);
        }

        return (IFilter)this.factory(serviceProvider, Arguments);
    }
}
復制代碼

三、結語

  相信看過上一篇文章的朋友都注意到了ServiceProviderActivatorUtilities 的不同,本文中的ServiceFilterAttribute和 TypeFilterAttribute 原理上也是通過它們來創建Filter的,所以使用場景就看大家如何來使用。其實最近看.NET Core的源代碼,看到的到處都是接口、工廠使用依賴注入形成擴展點的例子,其實微軟以前代碼的擴展點也挺多的,只是API並不那么開放,ASP.NET Core中我們看到了一個"開放"的框架。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM