ABP - 依賴註入(2)

来源:https://www.cnblogs.com/wewant/archive/2023/05/30/17125223.html
-Advertisement-
Play Games

# 依賴註入的使用 ## 構造方法註入 這是將服務註入類的最常用方法,是將依賴項註入類的首選方式,也是微軟推崇的模式。這樣,除非提供了所有構造方法註入的依賴項,否則無法構造類,顯示的聲明瞭類必需的服務,使開發人員一目瞭然。 ```csharp public class BookAppService ...


依賴註入的使用

構造方法註入

這是將服務註入類的最常用方法,是將依賴項註入類的首選方式,也是微軟推崇的模式。這樣,除非提供了所有構造方法註入的依賴項,否則無法構造類,顯示的聲明瞭類必需的服務,使開發人員一目瞭然。

public class BookAppService : ApplicationService
{
    private readonly IBookRepository _bookRepository;
		public TaxAppService(IBookRepository bookRepository)
    {
        _bookRepository = bookRepository;
    }
		public void DoSomething()
    {
        //...使用 _bookRepository...
    }
}

屬性註入

Microsoft依賴註入庫不支持屬性註入, 屬性註入是Abp框架使用autofac容器替換了微軟預設的容器之後才能使用的,屬於autofac的特性。

public class BookAppService : ITransientDependency
{
	public ILogger<BookAppService> Logger { get; set; }
	public BookAppService()
	{
        Logger = NullLogger<BookAppService>.Instance;
	}
	
	public void DoSomething()
	{
        //...使用 Logger 寫日誌...
	}
}

屬性註入依賴項通常被視為可選依賴項.這意味著不是必須的,沒有它們服務也可以正常工作.Logger就是這樣的依賴項,BookAppService可以繼續工作而無需日誌記錄。為了使依賴項成為可選的,我們通常會為依賴項設置預設值。一般是介面的一個空實現,內部不做任何操作。

屬性註入的一個限制是你不能在構造函數中使用依賴項,因為它是在對象構造之後設置的.

從IServiceProvider解析服務

public class MyService : ITransientDependency
{
    private readonly IServiceProvider _serviceProvider;
	public MyService(IServiceProvider serviceProvider)
    {
        _serviceProvider = serviceProvider;
    }
	
	public void DoSomething()
    {
        
		using (var scope = _serviceProvider.CreateScope())
		{
			var service1 = scope.ServiceProvider.GetService<IMyService1>();
			// …
		}
	}
}

通過GetServiceGetRequiredService可以直接從容器中解析出我們需要的服務,這兩個方法有一定的區別,推薦使用GetRequiredService方法。

這種情況下一般會使用using (var scope = _serviceProvider.CreateScope()){} 創建一個域,以便解析出來的服務能夠安全的被釋放,這也能夠減少一些記憶體消耗。

要註意的一個點是,在以前的ASP.NET Boilerplate 中服務的手動解析使用是通過IocManager的,這是靜態類,通過IocManager.Instance可以獲得容器對象,很方便使用。但是在現在的ABP中已經將這種方式移除了,以至於有時候在靜態類中需要使用一些服務的時候很不方便(雖然這種方式不是很正確,但是確實有時候會有這樣子的需求)。

所以在項目中如果真的有需要的話,可以自己構建一個類似於IocManager的方式,但是不推薦這樣子用,因為在某些情況確實會存在一些問題,導致一些資源和生命周期的衝突,特別是在單元測試之中。

/// <summary>
/// ServiceManager構建一個單例,用於寄放IServiceProvider
/// 使用此類時,需要在應用啟動的時候,調用Init方法
/// 一般情況下不推薦使用
/// </summary>
public class ServiceManager
{
	private ServiceManager()
	{
	}

	public static IServiceProvider ServiceProvider { get; private set; }

	public static void Init(IServiceProvider service)
	{
		ServiceProvider = service;
	}
}
public override void OnApplicationInitialization(ApplicationInitializationContext context)
{
	var app = context.GetApplicationBuilder();
	// 保存全局Ioc
	if (app != null)
	{
		ServiceManager.Init(app.ApplicationServices);
	}
}

Autofac的使用

容器替換

Abp框架中使用autofac依賴註入框架替換了微軟預設的容器,在volo.abp.autofac模塊中按照模塊化設計的思想進行了稍微的封裝,提供了便捷的替換方式。

使用autofac替換預設容器的方式:

(1) 在startup類中的ConfigureServices()中,在添加Abp的依賴註入時,通過選項替換。

public IServiceProvider ConfigureServices(IServiceCollection services)
{
    services.AddApplication<MyWebModule>(options =>
    {
        //Integrate Autofac!
        options.UseAutofac();
    });
    return services.BuildServiceProviderFromFactory();
}

(2) 在program.cs中,在.net Core 主機(host)創建的時候,調用userAutofac()擴展方法。

internal static IHostBuilder CreateHostBuilder(string[] args) =>
	            Host.CreateDefaultBuilder(args)
	                .ConfigureWebHostDefaults(webBuilder =>
	                {
	                    webBuilder.UseStartup<Startup>();
	                })
                .UseAutofac();

攔截器與動態代理

Abp框架使用autofac替換微軟預設容器的目的,主要就是為了利用autofac的攔截器功能,這是一種動態aop的實現方式,可以動態的攔截往容器中註冊的類,做一些想要的操作。abp框架中的很多橫切關註點都是通過這種方式實現的,如許可權認證,審計等。

在模塊類的PreConfigureServices()中,可以使用OnRegistred()方法,註冊一個回調,每次有類向容器中註冊時,都會調用這個回調。在回調中,我們可以對當前註冊的類型進行判斷(一般通過介面、特性等),如果滿足我們的條件,就添加一個攔截器。

image

image

image

在我們調用useAutofac()時,內部會調用UseServiceProviderFactory()創建了一個AbpAutofacServiceProviderFactory,這是實現了微軟IServiceProviderFactory<>介面的一個容器工廠,任何容器提供了這個介面實現類,並且按照約定提供了ContainerBuilder,都可以作為.net core中的依賴註入容器。

在.net Core應用程式啟動的時候會調用IServiceProviderFactory<>介面的CreateBuilder(IServiceCollection services)方法,我們註冊的回調函數就是在這個時候執行的。遍歷每一個類,在多個回調中進行執行,符合條件的都會添加一個攔截器。

image

image

最終是向autofac容器註冊攔截器。

攔截器類需要實現IAbpInterceptor介面,或者繼承AbpInterceptor,實現其中的InterceptAsync方法。如果對某一個類配置了攔截器,實際上它在註冊到容器中時,abp會基於攔截器動態生成一個代理類,再將代理類註冊到autofac中。

動態代理是基於Castle實現的。abp繼承了Castle動態代理的AsyncDeterminationInterceptor,通過繼承了AsyncInterceptorBase使用適配器模式將攔截器進行包裝,並且將方法執行的參數進行封裝傳遞到我們定義的攔截器中。

image

image

執行被攔截的類的方法時,實際上通過代理類在執行InterceptAsync()方法,這類似於傳統過濾器的效果,所以通過這種方式,我們可以在方法執行前後添加一些和業務無關的全局的操作,即一些橫切關註點。



ABP 系列總結:

目錄:ABP 系列總結
上一篇:ABP - 依賴註入(1)


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

-Advertisement-
Play Games
更多相關文章
  • 242. 有效的字母異位詞 ```cpp class Solution { public: bool isAnagram(string s, string t) { if(s.size()!=t.size()) return false; int ans[26]={0}; for(auto& ch: ...
  • 前言 咱換工作啦! 新工作這邊需要用到的開發語言是 Haxe,最近大概會寫幾篇筆記。Haxe 的介紹就不寫了,打算記錄點有用的學習內容,先從搭建開發環境開始吧! 當前適用版本: VSCode:Current Latest Version Haxe 版本:4.3.1 文章最近更新日期:2023.05. ...
  • 在最近的互聯網項目開發中,需要獲取用戶的訪問ip信息,併進行後續統計分析。 這些ip信息是在第三方的服務中分組存放的,且每個分組都都是分頁(1頁10條)存放的,如果一次性訪問大量的數據,API很有可能會報錯。 怎樣通過HTTP的方式去獲取到信息,並且模擬瀏覽器每頁每頁獲取10條的信息,且持久到資料庫... ...
  • 如前所述,在前幾章內容中筆者簡單介紹了`記憶體讀寫`的基本實現方式,這其中包括了`CR3切換`讀寫,`MDL映射`讀寫,`記憶體拷貝`讀寫,本章將在如前所述的讀寫函數進一步封裝,並以此來實現驅動讀寫記憶體浮點數的目的。記憶體`浮點數`的讀寫依賴於`讀寫記憶體位元組`的實現,因為浮點數本質上也可以看作是一個位元組集... ...
  • # 還在愁個人博客沒有圖片放?🥱 # 前言 之前將爬取的圖片下載後,我也是放到我了我的博客上面 [ZY知識庫 · ZY - Home Page (pljzy.top)](https://pljzy.top/) 然後順便寫了一個隨機返回圖片的介面,現在我就來說一下如何使用這個介面,以便在自己的博客上 ...
  • 在 WPF 中,CanContentScroll 是 ScrollViewer 控制項的一個附加屬性,它控制滾動視圖中的內容是否按項或像素來滾動。 當 CanContentScroll 設置為 false 時,表示 ScrollViewer 控制項使用逐像素的滾動方式,這意味著滾動視圖中的內容會以像素為 ...
  • 1. 泛型 泛指某種類型。 1、使用參數形式定義 2、使用時傳入具體類型 3、編譯時檢查類型安全 4、邏輯上是多個不同類型 泛型與非泛型之間的區別 性能高:可以避免裝箱和拆箱操作 類型安全 :在進行類型轉換的時候不會拋出異常 代碼重用:定義一次,用許多種不同類型實例化 代碼擴展性好 ArrayLis ...
  • 編寫包含多個 `csproj` 的程式時,隨著項目數量的持續增加,可能涉及一些文件夾的變動,手動添加項目或者變動會變得非常麻煩,這個時候,可以利用 `dotnet cli` 幫助我們完成。 如果從零開始,我們可以新建一個解決方案。 ```powershell dotnet new sln -n to ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...