談一談autofac組件的實(shí)例范圍
實(shí)例范圍決定如何在請(qǐng)求之間共享服務(wù)。
原文地址:http://docs.autofac.org/en/latest/lifetime/instance-scope.html
每個(gè)依賴(lài)一個(gè)實(shí)例
使用這個(gè)選項(xiàng),每次請(qǐng)求服務(wù)都會(huì)返回一個(gè)新實(shí)例。使用 InstancePerDependency() 指定。這是默認(rèn)選項(xiàng)。下面的代碼,第2行和第3行是等價(jià)的。
var builder = new ContainerBuilder(); builder.RegisterType<Worker>(); builder.RegisterType<Worker>().InstancePerDependency();
下面的代碼,每次循環(huán)都生成一個(gè)新的實(shí)例,一共生成 100 個(gè)實(shí)例。
using(var scope = container.BeginLifetimeScope()) { for(var i = 0; i < 100; i++) { var w = scope.Resolve<Worker>(); w.DoWork(); } }
單個(gè)實(shí)例
使用這個(gè)選項(xiàng),在根范圍或嵌套范圍中請(qǐng)求服務(wù),都返回同一個(gè)的實(shí)例。使用 SingleInstance() 指定。
var builder = new ContainerBuilder();
builder.RegisterType<Worker>().SingleInstance();
下面的代碼,w1 和 w2 始終是同一個(gè)對(duì)象,100 次循環(huán)只有一個(gè) Worker 類(lèi)的實(shí)例。
using(var scope1 = container.BeginLifetimeScope()) { for(var i = 0; i < 100; i++) { var w1 = scope1.Resolve<Worker>(); using(var scope2 = scope1.BeginLifetimeScope()) { var w2 = scope2.Resolve<Worker>(); } } }
每個(gè)生命周期范圍一個(gè)實(shí)例
使用這個(gè)選項(xiàng),在特定的 ILifetimeScope 中請(qǐng)求服務(wù),只返回一個(gè)實(shí)例。使用 InstancePerLifetimeScope() 指定。下面的代碼中,scope1 中的 100 次 w1 是同一個(gè)對(duì)象,scope2 中的 100 次 w2 是同一個(gè)對(duì)象,但是 w1 和 w2 不是同一個(gè)對(duì)象。
var builder = new ContainerBuilder(); builder.RegisterType<Worker>().InstancePerLifetimeScope(); using(var scope1 = container.BeginLifetimeScope()) { for(var i = 0; i < 100; i++) { var w1 = scope1.Resolve<Worker>(); } } using(var scope2 = container.BeginLifetimeScope()) { for(var i = 0; i < 100; i++) { var w2 = scope2.Resolve<Worker>(); } }
每個(gè)匹配的生命周期范圍一個(gè)實(shí)例
類(lèi)似于上面【每個(gè)生命周期范圍一個(gè)實(shí)例】,但可以提供更多控制。使用此選項(xiàng),允許為 ILifetimeScope 對(duì)象提供“標(biāo)記”。在標(biāo)記匹配的范圍中只有一個(gè)實(shí)例。使用 InstancePerMatchingLifetimeScope() 方法指定。
var builder = new ContainerBuilder(); builder.RegisterType<Worker>().InstancePerMatchingLifetimeScope("x");
下面的代碼中,w1 和 w2 相同,w3 和 w4 相同,但 w1 和 w3 不同。
using(var scope1 = container.BeginLifetimeScope("x")) { for(var i = 0; i < 100; i++) { var w1 = scope1.Resolve<Worker>(); using(var scope2 = scope1.BeginLifetimeScope()) { var w2 = scope2.Resolve<Worker>(); } } } using(var scope3 = container.BeginLifetimeScope("x")) { for(var i = 0; i < 100; i++) { var w3 = scope3.Resolve<Worker>(); using(var scope4 = scope1.BeginLifetimeScope()) { var w4 = scope4.Resolve<Worker>(); } } }
解析時(shí)必須提供提供合適的標(biāo)記,以下代碼會(huì)拋出異常。
using(var noTagScope = container.BeginLifetimeScope()) { var fail = noTagScope.Resolve<Worker>(); }
每個(gè)請(qǐng)求一個(gè)實(shí)例
有些應(yīng)用程序天然具有【請(qǐng)求】語(yǔ)義,例如 ASP.NET MVC 或 WebForm 應(yīng)用程序?!久總€(gè)請(qǐng)求一個(gè)實(shí)例】在【每個(gè)匹配的生命周期范圍一個(gè)實(shí)例】基礎(chǔ)上,通過(guò)提供范圍標(biāo)記,注冊(cè)函數(shù)和常見(jiàn)類(lèi)型集成實(shí)現(xiàn)。本質(zhì)上是【每個(gè)匹配的生命周期范圍一個(gè)實(shí)例】。
var builder = new ContainerBuilder(); builder.RegisterType<Worker>().InstancePerRequest();
ASP.NET Core 使用【每個(gè)生命周期范圍一個(gè)實(shí)例】,而不是【每個(gè)請(qǐng)求一個(gè)實(shí)例】。
每個(gè)Owned 一個(gè)實(shí)例
Owned<T> 隱式關(guān)聯(lián)類(lèi)型創(chuàng)建嵌套的生命周期范圍。使用 instance-per-owned 注冊(cè),可將依賴(lài)限定在 owned 實(shí)例中。
var builder = new ContainerBuilder(); builder.RegisterType<MessageHandler>(); builder.RegisterType<ServiceForHandler>().InstancePerOwned<MessageHandler>();
本例中 ServiceForHandler 服務(wù)會(huì)限制在 MessageHandler 實(shí)例范圍內(nèi)。
using(var scope = container.BeginLifetimeScope()) { // MessageHandler 和附屬的 ServiceForHandler // 在 scope 下面的一個(gè)微型的 lifetime scope 中。 // 解析 Owned<T> 需要程序員負(fù)責(zé)執(zhí)行清理工作。 var h1 = scope.Resolve<Owned<MessageHandler>>(); h1.Dispose(); }
線(xiàn)程范圍
Autofac 可以強(qiáng)制使A線(xiàn)程的對(duì)象不滿(mǎn)足B線(xiàn)程的依賴(lài)。
var builder = new ContainerBuilder(); builder.RegisterType<MyThreadScopedComponent>() .InstancePerLifetimeScope(); var container = builder.Build();
然后讓每個(gè)創(chuàng)建自己的 lifetime scope
void ThreadStart() { using (var threadLifetime = container.BeginLifetimeScope()) { var thisThreadsInstance = threadLifetime.Resolve<MyThreadScopedComponent>(); } }
重要:在多線(xiàn)程場(chǎng)景下,要小心不要將父范圍清理掉。否則,派生線(xiàn)程中的子范圍將無(wú)法解析服務(wù)。
每個(gè)線(xiàn)程都將有自己的 MyThreadScopedComponent 實(shí)例,本質(zhì)上是生命周期范圍內(nèi)的單例。范圍內(nèi)的實(shí)例不會(huì)提供到外部,因此很容易保持線(xiàn)程間的組件隔離。
通過(guò)添加 ILifetimeScope 參數(shù),可將父范圍注入到生成線(xiàn)程的代碼中,Autofac 會(huì)將當(dāng)前范圍自動(dòng)注入,接下來(lái)可以使用它創(chuàng)建嵌套范圍。
public class ThreadCreator { private ILifetimeScope _parentScope; public ThreadCreator(ILifetimeScope parentScope) { this._parentScope = parentScope; } public void ThreadStart() { using (var threadLifetime = this._parentScope.BeginLifetimeScope()) { var thisThreadsInstance = threadLifetime.Resolve<MyThreadScopedComponent>(); } } }
如果想進(jìn)一步控制,可以使用【每個(gè)匹配的生命周期范圍一個(gè)實(shí)例】將線(xiàn)程范圍的組件同內(nèi)部生命周期范圍關(guān)聯(lián)起來(lái),這個(gè)方式的效果如圖:
圖中的"Context"是 BeginLifetimeScope 方法創(chuàng)建的生命周期范圍。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- ASP.Net Core基于EF6、Unitwork、Autofac實(shí)現(xiàn)Repository模式
- C#?利用Autofac批量接口注入依賴(lài)的問(wèn)題小結(jié)
- .net6引入autofac框架
- .NET中IoC框架Autofac用法講解
- .Net Core 之AutoFac的使用
- .Net Core中使用Autofac替換自帶的DI容器的示例
- C# 使用 Castle 實(shí)現(xiàn) AOP及如何用 Autofac 集成 Castle
- NET Core 3.0 AutoFac內(nèi)置DI替換的新姿勢(shì)分享
- 淺析依賴(lài)注入框架Autofac的使用
- .NET?Core使用Autofac容器的DI依賴(lài)注入,IOC控制反轉(zhuǎn)及AOP切面編程
相關(guān)文章
C# 使用 Castle 實(shí)現(xiàn) AOP及如何用 Autofac 集成 Castle
這篇文章主要介紹了C# 使用 Castle 實(shí)現(xiàn) AOP及如何用 Autofac 集成 Castle,幫助大家更好的理解和學(xué)習(xí)使用c#,感興趣的朋友可以了解下2021-02-02C#難點(diǎn)逐個(gè)擊破(9):類(lèi)型轉(zhuǎn)換
類(lèi)型之間的轉(zhuǎn)換可以分為隱式轉(zhuǎn)換與顯式轉(zhuǎn)換,如int類(lèi)型可直接轉(zhuǎn)換為long類(lèi)型。2010-02-02C# 如何實(shí)現(xiàn)一個(gè)基于值相等性比較的字典
這篇文章主要介紹了C# 如何實(shí)現(xiàn)一個(gè)基于值相等性比較的字典,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下2021-02-02Unity3D開(kāi)發(fā)教程:憤怒的小鳥(niǎo)
這篇文章詳細(xì)的講解了如何從0開(kāi)發(fā)出一個(gè)Unity3D的小游戲憤怒的小鳥(niǎo),本文包含大量的圖片與文字描述,也含有大量的源代碼,可以讓你快速入手,希望本篇文章對(duì)你有所幫助2021-06-06C#實(shí)現(xiàn)ProperTyGrid自定義屬性的方法
這篇文章主要介紹了C#實(shí)現(xiàn)ProperTyGrid自定義屬性的方法,主要通過(guò)接口ICustomTypeDescriptor實(shí)現(xiàn),需要的朋友可以參考下2014-09-09C# 無(wú)需COM組件創(chuàng)建快捷方式的實(shí)現(xiàn)代碼
做一個(gè)小程序, 需要?jiǎng)?chuàng)建快捷方式, 網(wǎng)上普遍的做法是引入 COM 組件, 雖然也挺方便的, 但引入之后, 程序就需要多帶一個(gè) dll 文件, 這樣, 想做成單文件便攜版就不行了2011-05-05不能在子類(lèi)或外部類(lèi)發(fā)布C#事件代碼分析
一個(gè)朋友問(wèn)了一個(gè)問(wèn)題:“為什么不能在子類(lèi)或外部發(fā)布C#事件?”,我說(shuō)我不知道,要看看生產(chǎn)的IL代碼,下面我們看看2013-11-11