dotNET Core 3.X 使用 Autofac 來增強依賴註入

来源:https://www.cnblogs.com/oec2003/archive/2020/06/08/13069058.html
-Advertisement-
Play Games

在上一篇《dotNET Core 3.X 依賴註入》中簡單介紹了 dotNET Core 框架本身的依賴註入功能,大部分情況下使用框架的依賴註入功能就可以滿足了,在一些特殊場景下,我們就需要引入第三方的註入框架。 為什麼要使用 Autofac? 如果您在之前的 dotNET Framwork 時代使 ...


在上一篇《dotNET Core 3.X 依賴註入》中簡單介紹了 dotNET Core 框架本身的依賴註入功能,大部分情況下使用框架的依賴註入功能就可以滿足了,在一些特殊場景下,我們就需要引入第三方的註入框架。

為什麼要使用 Autofac?

如果您在之前的 dotNET Framwork 時代使用過依賴註入,那麼對 Autofac 一定不會陌生,在 dotNET Core 中也可以很方便的使用 Autofac,之所以使用第三方註入框架,是因為能提供更多的功能:

  • 屬性註入
  • 批量註入
  • 動態代理的 AOP 功能

在 dotNET Core 中使用 Autofac

在 dotNET Core 2.x 和 3.x 中使用 Autofac 是有區別的,所以下麵分別介紹在兩個版本中的簡單使用。

2.x

1、創建 dotNET Core 2.1 版本的 WebAPI 項目;
2、創建 IUserService 介面和 UserService 類

public interface IUserService
{
    string GetUserName();
}
public class UserService: IUserService
{
    public string GetUserName()
    {
        return "oec2003";
    }
}

3、創建 UserController,在構造函數中添加依賴註入

[Route("api/[controller]/[action]")]
[ApiController]
public class UserController: ControllerBase
{
    private readonly IUserService _userService;

    public UserController(IUserService userService)
    {
        _userService = userService;
    }
    public string GetUserName()
    {
        return _userService.GetUserName();
    }
}

4、添加 Autofac.Extensions.DependencyInjection 的 NuGet 引用

5、修改 Startup 類的 ConfigureServices 方法

public IServiceProvider ConfigureServices(IServiceCollection services)
{
    services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_1);
    //創建 Autofac 容器
    var containerBuilder = new ContainerBuilder();
    containerBuilder.Populate(services);
    //將 UserService 類作為 IUserService 的實現進行註冊
    containerBuilder.RegisterType<UserService>().As<IUserService>().InstancePerLifetimeScope();
    var container = containerBuilder.Build();
    //接管內置的容器
    return new AutofacServiceProvider(container);
}

3.x

1、創建 dotNET Core 3.x 的項目和相關類,參考上面的一到四步;

2、修改 Program 類,使用 AutofacServiceProviderFactory 來替代創建服務提供程式的工廠:

public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .UseServiceProviderFactory(new AutofacServiceProviderFactory())
            .ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });

3、修改 Startup 類,在該類中添加 ConfigureContainer 方法,和ConfigureServices 方法一樣,框架也是通過命名約束來進行執行的:

public void ConfigureContainer(ContainerBuilder builder)
{
    builder.RegisterType<UserService>().As<IUserService>().InstancePerLifetimeScope();
}

Autofac 的增強功能

下麵的所有示例全部在 dotNET Core 3.1 版本中完成。

屬性註入

dotNET Core 框架本身的依賴註入只支持構造函數和 FromSerice 的方式,Autofac 可以支持屬性的註入。

使用屬性註入很簡單,在註冊類型時調用 PropertiesAutowired 方法即可,具體步驟如下:

1、調整 UserController ,以屬性的方式來定義 IUserService

public class UserController: ControllerBase
{
    public IUserService UserService { get; set; }
    
    public string GetUserName()
    {
        return UserService.GetUserName();
    }
}

2、修改 Startup 類的 ConfigureServices 方法,添加 AddControllersAsServices 方法的調用

public void ConfigureServices(IServiceCollection services)
{
    services.AddControllers()
        .AddControllersAsServices();
}

3、修改 Startup 類的 ConfigureContainer ,

public void ConfigureContainer(ContainerBuilder builder)
{
    builder.RegisterType<UserService>().As<IUserService>()
        .InstancePerLifetimeScope();

    var controllerBaseType = typeof(ControllerBase); 
    builder.RegisterAssemblyTypes(typeof(Program).Assembly)
        .Where(t => controllerBaseType.IsAssignableFrom(t) && t != controllerBaseType)
        .PropertiesAutowired();
}
  • 只要在 Controller 中需要做屬性註入的時候,才需要在 ConfigureServices 方法中添加對 AddControllersAsServices 方法的調用;
  • PropertiesAutowired 方法添加在使用屬性的註入類型中,比如上面代碼是在 Controller 中使用屬性,所以 PropertiesAutowired 添加對所有 Controller註冊的後面;
  • 如果在 UserService 類以屬性的方式對 IDeptService 引用,註冊的方式如下:
public void ConfigureContainer(ContainerBuilder builder)
{
    builder.RegisterType<DeptService>().As<IDeptService>()
        .InstancePerLifetimeScope();
    builder.RegisterType<UserService>().As<IUserService>()
        .PropertiesAutowired()
        .InstancePerLifetimeScope();
}

批量註冊

其實上面的代碼中已經涉及到了批量註冊,就是對所有的 Controller 進行註冊:

var controllerBaseType = typeof(ControllerBase); 
builder.RegisterAssemblyTypes(typeof(Program).Assembly)
    .Where(t => controllerBaseType.IsAssignableFrom(t) && t != controllerBaseType)
    .PropertiesAutowired();
  • 所有的 Controller 都是繼承自基類 ControllerBase,先獲取基類的類型;
  • 找到 Program 類所在的程式集中所有實現了 ControllerBase 的類型進行註冊。

再來看另一種情況,上面例子中創建 UserServicce 服務,現在再創建 DeptService 服務類:

public interface IDeptService
{
    string GetDeptName();
}
public class DeptService:IDeptService
{
    public string GetDeptName()
    {
        return "產品部";
    }
}

修改 Startup 類的 ConfigureContainer 方法來實現批量註冊:

public void ConfigureContainer(ContainerBuilder builder)
{
    builder.RegisterAssemblyTypes(typeof(Program).Assembly)
        .Where(t => t.Name.EndsWith("Service"))
        .AsImplementedInterfaces()
        .InstancePerLifetimeScope();
}

找到 Program 類所在的程式集中所有以 Service 命名的類型進行註冊。更多的情況就根據實際場景舉一反三了。

動態代理的 AOP 功能

使用動態代理的功能,需要引用 NuGet 包:Autofac.Extras.DynamicProxy,如下圖:

AOP 的概念這裡就不在贅述,和 dotNET Core 內置的攔截器(Filter、中間件)的區別是 Autofac 的 AOP 基於業務方法而不是 HTTP。

1、創建 UserServiceInterceptor 攔截類,繼承自 IInterceptor

public class UserServiceInterceptor:IInterceptor
{
    public virtual void Intercept(IInvocation invocation)
    {
        Console.WriteLine($"{DateTime.Now}: 方法執行前");
        invocation.Proceed();
        Console.WriteLine($"{DateTime.Now}: 方法執行後");
    }
}

2、修改 Startup 類中的 ConfigureContainer 方法,進行 AOP 的註冊

public void ConfigureContainer(ContainerBuilder builder)
{
    builder.RegisterType<UserServiceInterceptor>();
    builder.RegisterType<UserService>().As<IUserService>()
        .EnableInterfaceInterceptors()
        .InstancePerLifetimeScope();
}
  • 註冊 UserServiceInterceptor 攔截器
  • 註冊 UserService 服務的時候調用 EnableInterfaceInterceptors 啟用攔截器

3、修改 UserService 類,添加 AOP 特性標記

[Intercept(typeof(UserServiceInterceptor))]
public class UserService: IUserService
{
    //public IDeptService DeptService { get; set; }
    public string GetUserName()
    {
        Console.WriteLine($"{DateTime.Now}: 方法執行中");
        return "oec2003";
        //return $"oec2003({DeptService.GetDeptName()})";
    }
}

4、調用結果如下:

總結

本文算是拋磚引入,Autofac 還有許多的功能由於目前沒有使用到,也就沒有放到本文中,比如子容器等。具體使用 dotNET Core 框架自身的依賴註入,還是使用 Autofac,要看具體的場景了,當然兩者也是可以並存的。

示例代碼:https://github.com/oec2003/DotNetCoreThreeAPIDemo/tree/master/AutofacNetCore3.1Demo


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 21 反射 21.1 反射概念 反射主要是指程式可以訪問、檢測和修改其本身狀態或行為的一種能力。在Python中最重要的4個方法如下所示: getattr:獲取指定字元串名稱的對象屬性 setattr:給對象添加類屬性 hasattr:判斷對象是否存在某個對應的對象 delattr:刪除指定的屬性 ...
  • 以前對於這兩個概念其實沒有多少認識,只知道註解是標記,對於反射的印象就是類比於游戲進度的回顯吧,慚愧慚愧~~~學習後有了一些更深入的瞭解,記錄一下,加深印象。 註解(Annotation) 什麼是註解? 是 JDK5.0 引入的一種註釋機制,代碼輔助工具,由內置註解和元註解組成。核心作用是對類、方法 ...
  • 項目簡介 項目來源於:https://github.com/SuperiorNature/Java-Enterprise-electronic-mall 本系統是基於JSP+Structs+JDBC+Mysql實現的電子商城系統。界面和技術都比較過時,適合structs入門者。 難度等級:中等 技術 ...
  • 用戶認證 Django 內置一個 auth 模塊,幫助用戶實現註冊、登錄、註銷以及修改密碼等功能,幫助開發者省去了很多功夫 用於認證的數據表 auth_user User是auth模塊中維護用戶信息的關係模式(繼承了models.Model), 資料庫中該表被命名為auth_user auth_gr ...
  • 大綱 學前必讀 學習不能快速成功,但一定可以快速入門整體課程思路:1.實踐為主,理論化偏少2.課程筆記有完整的案例和代碼,(為了學習效率)再開始之前我會簡單粗暴的介紹知識點案例思路, 有基礎的同學聽了之後可以直接結合筆記寫代碼, 如果沒聽懂再向下看視頻,我會手把手編寫代碼和演示測試結果;3.重要提示 ...
  • 本篇主要學習如何Python自定義模塊並調用該模塊,並重點介紹Python正則表達式的強大的文本處理能力。 案例故事: 任何一款終端產品只要涉及音頻輸出,就肯定涉及音頻的解碼, 作為一名專業的AV (Audio & Video)測試人員,我們需要一堆的規範化標準的的音頻測試文件, 但是發現音頻資源名 ...
  • 如果基類有友元類或友元函數,則其派生類不會因繼承關係而也有此友元類或友元函數。如果基類是某類的友元,則這種友元關係是被繼承的。即被派生類繼承過來的成員函數,如果原來是某類的友元函數,那麼它作為派生類的成員函數仍然是某類的友元函數。總之,基類的友元不一定是派生類的友元;基類的成員函數是某類的友元函數,... ...
  • Spring 框架提供了許多介面,可以使用這些介面來定製化 bean ,而非簡單的 getter/setter 或者構造器註入。細翻 Spring Cloud Netflix、Spring Cloud Alibaba 等這些構建在 Spring Framework 的成熟框架源碼,你會發現大量的擴展... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...