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
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...