ASP.NET Core依賴註入之旅:針對服務註冊的驗證

来源:https://www.cnblogs.com/green-jcx/archive/2022/08/09/16566740.html
-Advertisement-
Play Games

1.避免Scoped模式註冊的服務變成Singleton模式 當提供一個生命周期模式為Singleton的服務實例時,如果發現該服務中還依賴生命周期模式為Scoped的服務實例(Scoped服務實例將被一個Singleton服務實例所引用),那麼這個被依賴的Scoped服務實例最終會成為一個Sing ...


1.避免Scoped模式註冊的服務變成Singleton模式

當提供一個生命周期模式為Singleton的服務實例時,如果發現該服務中還依賴生命周期模式為Scoped的服務實例(Scoped服務實例將被一個Singleton服務實例所引用),那麼這個被依賴的Scoped服務實例最終會成為一個Singleton模式的服務實例。這是因為提供Singleton服務的容器是根容器,Scoped服務間接的被根容器所創建提供了,如果Scoped服務由子容器進行提供,那麼Singleton和Scoped這兩種生命周期模式才會產生差別。

在ASP.NET Core應用中,將某個服務註冊的生命周期設置為Scoped,其意圖是希望Scoped服務實例的創建和釋放是作用於某個HTTP請求範圍內的。如果不註意,將Scoped服務實例引用到了Singleton服務實例中,對於這種情況Scoped和Singleton的服務實例沒有區別的。這樣的Scoped服務實例直到應用關閉才會被釋放,這無疑違背我們使用Scoped模式的初衷。這種“混淆”如果沒有察覺到,可能會在實際的應用中造成難以估量的後果,例如在Singleton服務中引用的Scoped服務是一個資料庫連接對象,這會導致資料庫長時間連接沒有及時釋放,從而導致程式出現異常。

為了避免Scoped模式註冊的服務“隱式”的變成Singleton模式的服務帶來的風險,.NET Core為我們提供了一種驗證方式來規避這樣的行為,這個方式就是將ServiceProviderOptions配置對象的ValidateScopes屬性設置為True。

當開啟了這個驗證後,依賴註入框架則會對註冊Scoped模式的服務進行檢查,確保不會出現如下情況:

  • 有根容器去提供Scoped的服務實例;
  • Singleton服務中存在對Scoped服務的依賴;

一旦開啟針對Scoped模式服務的註冊驗證,如果存在以上的兩種情況,那麼程式啟動時會拋出異常。

ValidateScopes的值在開發環境下預設值是為True,為了確保在生產環境或其他環境始終開啟驗證,我們可以在Program類的CreateHostBuilder方法中配置ServiceProviderOptions對象。

該驗證方式被官方稱作為“服務範圍”的驗證,並建議應用程式開啟此驗證,以確保我們註冊Scoped模式的服務僅作用於某個服務範圍,而不會“悄悄地”演變成作用於整個應用程式範圍的Singleton模式。


2.驗證服務註冊是否能夠提供相應的實例

依賴註入框架中進行服務註冊的信息一般都存放於ServiceDescriptor的對象中,而容器對象就是根據ServiceDescriptor對象中的註冊信息進行服務實例的提供。ServiceProviderOptions配置類型除了用於針對“服務範圍”的驗證ValidateScopes屬性之外,還有一個ValidateOnBuild屬性。如果將該屬性設置為True,這就意味著容器對象在構建時,會對每個ServiceDescriptor對象中的註冊信息實施有效性驗證,如果服務註冊信息不能提供出對應的實例則會拋出異常。

使用ValidateOnBuild屬性進行驗證的目的是因為,往往有些服務能夠正常註冊但不代表,容器能夠根據註冊信息成功的提供實例。下麵我將通過一個代碼示例來印證這一情況,並演示使用ValidateOnBuild屬性進行驗證的方式。

 1 using Microsoft.Extensions.DependencyInjection;
 2 using System;
 3 using System.Diagnostics;
 4 
 5 namespace ConsoleApp1
 6 {
 7 
 8     public interface IFooBar { }
 9     public class FooBar : IFooBar
10     {
11         private FooBar() { }
12     }
13 
14     internal class Program
15     {
16         static void Main(string[] args)
17         {
18             Console.WriteLine("ValidateOnBuild的值為True:");
19             BuildServiceProvider(true);
20             Console.WriteLine();
21 
22             Console.WriteLine("ValidateOnBuild的值為False:");
23             BuildServiceProvider(false);
24 
25         } // END Main()
26 
27         static void BuildServiceProvider(bool validateOnBuild)
28         {
29             try
30             {
31                 var options = new ServiceProviderOptions { ValidateOnBuild = validateOnBuild };
32              var provider=   new ServiceCollection()
33                     .AddSingleton<IFooBar, FooBar>()
34                     .BuildServiceProvider(options);
35                 Console.WriteLine($"程式運行正常;");
36             }
37             catch (Exception e)
38             {
39                 Console.WriteLine($"程式出現異常;異常信息:{e.Message};");
40             }
41 
42         } // END BuildServiceProvider()
43 
44 
45     }
46 }

在上面的代碼示例中,服務註冊時指定的實現類型為FooBar,而該類型中唯一的構造函數是一個私有的。我們都知道創建對象的實例,則必須要調用對象類型的構造函數,而FooBar的構造函數是私有的,無法對外界所調用,則也意味著無法創建相應的實例。

運行上面的代碼示例後我們會發現,對於ValidateOnBuild屬性值設置為False的情況,程式可以正常的執行服務註冊的方法,這種現象隱瞞了註冊時的錯誤(類型中存在私有構造函數),這種做法顯然會對後續使用到FooBar對象的程式功能帶來影響。而對於ValidateOnBuild屬性值設置為True的情況,程式則直接拋出了異常並且給出了詳細的錯誤信息,對於這種做法我們可以更好的規避服務註冊時所產生的風險。

 

本示例代碼是在“服務定位器模式”下配置的ValidateOnBuild屬性,這是為了更好的演示。當然,我們在實際的開發中通常是依賴註入的形式,那麼相應的方式是通過在CreateHostBuilder方法中ServiceProviderOptions對象進行配置,如下圖:

 


3.總結

本文介紹的關於“針對服務註冊的驗證”的主題,實際上就是介紹ServiceProviderOptions類型中ValidateScopesValidateOnBuild屬性的使用場景和方式。VailedateScopes屬性主要確保我們註冊的Scoped服務不會在某些情況下變成Singleton服務,ValidateOnBuild屬性主要用於驗證服務註冊信息是否能成功的提供出對應的服務示例。

知識改變命運
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 大家好,我是三友,我又來了~~ 最近仍然暢游在RocketMQ的源碼中,這幾天剛好翻到了消費者的源碼,發現RocketMQ的對於push消費方式的實現簡直太聰明瞭,所以趁著我腦子裡還有點印象的時候,趕緊來寫一篇文章,來掰扯一下,防止過兩天就忘得一干二凈了。 MQ消費方式 消費方式就是指消費者如何從M ...
  • .精華筆記: 1)成員內部類: 應用率不高 1.1)類中套類,外面的稱為外部類,裡面的稱為內部類 1.2)內部類通常只服務於外部類,對外不具備可見性 1.3)內部類對象通常在外部類中創建 1.4)內部類中可以直接訪問外部類的成員(包括私有的) 內部類中有一個隱式的引用,指向了創建它的外部類對象 外部 ...
  • Python有一個for...else語法,它的寫法如下 for i in range(0,100): if i == 3: break else: print("Not found") 該語句表示:若for迴圈遍歷完畢,則執行else部分的語句。也就是說上述代碼不會有任何輸出,而下述代碼會輸出“N ...
  • 精華筆記: static final常量:應用率高 必須聲明同時初始化 由類名打點來訪問,不能被改變 建議:常量所有字母都大寫,多個單詞用_分隔 編譯器在編譯時會將常量直接替換為具體的數,效率高 何時用:數據永遠不變,並且經常使用 抽象方法: 由abstract修飾 只有方法的定義,沒有具體的實現( ...
  • 一、實現原理 在Servlet3協議規範中,包含在JAR文件/META-INFO/resources/路徑下的資源可以直接訪問。 二、舉例說明 如下圖所示,是我新建的一個Spring Boot Starter項目:zimug-minitor-threadpool,用於實現可配置、可觀測的線程池。其中 ...
  • 1、我們的目標是獲取微博某博主的全部圖片、視頻 2、拿到網址後 我們先觀察 打開F12 隨著下滑我們發現載入出來了一個叫mymblog的東西,展開響應發現需要的東西就在裡面 3、重點來了!!! 通過觀察發現第二頁比第一頁多了參數since_id 而第二頁的since_id參數剛好在上一頁中能獲取到, ...
  • django2 路由控制器 Route路由,是一種映射關係。路由是把客戶端請求的url路徑和用戶請求的應用程式,這裡意指django裡面的視圖進行綁定映射的一種關係。 請求路徑和視圖函數不是一一對應的關係 在django中所有的路由最終都被保存到一個叫urlpatterns的文件里,並且該文件必須在 ...
  • 索引時資料庫提高數據查詢處理性能的一個非常關鍵的技術,索引的使用可以對性能產生上百倍甚至上千倍的影響。接下來,會介紹索引的基本原理、概念,並深入學習資料庫中所使用的索引結構和存儲方式,以及如何管理、維護索引等。 1.索引的基本概念 索引時用來快速查詢表記錄的一種存儲結構,一般使用索引有一下兩個方面: ...
一周排行
    -Advertisement-
    Play Games
  • JWT(JSON Web Token)是一種用於在網路應用之間傳遞信息的開放標準(RFC 7519)。它使用 JSON 對象在安全可靠的方式下傳遞信息,通常用於身份驗證和信息交換。 在Web API中,JWT通常用於對用戶進行身份驗證和授權。當用戶登錄成功後,伺服器會生成一個Token並返回給客戶端 ...
  • 老周在幾個世紀前曾寫過樹莓派相關的 iOT 水文,之所以沒寫 Nano Framework 相關的內容,是因為那時候這貨還不成熟,可玩性不高。不過,這貨現在已經相對完善,老周都把它用在項目上了——第一個是自製的智能插座,這個某寶上50多塊可以買到,搜“esp32 插座”就能找到。一種是 86 型盒子 ...
  • 引言 上一篇我們創建了一個Sample.Api項目和Sample.Repository,並且帶大家熟悉了一下Moq的概念,這一章我們來實戰一下在xUnit項目使用依賴註入。 Xunit.DependencyInjection Xunit.DependencyInjection 是一個用於 xUnit ...
  • 在 Avalonia 中,樣式是定義控制項外觀的一種方式,而控制項主題則是一組樣式和資源,用於定義應用程式的整體外觀和感覺。本文將深入探討這些概念,並提供示例代碼以幫助您更好地理解它們。 樣式是什麼? 樣式是一組屬性,用於定義控制項的外觀。它們可以包括背景色、邊框、字體樣式等。在 Avalonia 中,樣 ...
  • 在處理大型Excel工作簿時,有時候我們需要在工作表中凍結窗格,這樣可以在滾動查看數據的同時保持某些行或列固定不動。凍結窗格可以幫助我們更容易地導航和理解複雜的數據集。相反,當你不需要凍結窗格時,你可能需要解凍它們以獲得完整的視野。 下麵將介紹如何使用免費.NET庫通過C#實現凍結Excel視窗以鎖 ...
  • .NET 部署 IIS 的簡單步驟一: 下載 dotnet-hosting-x.y.z-win.exe ,下載地址:.NET Downloads (Linux, macOS, and Windows) (microsoft.com) .NET 部署 IIS 的簡單步驟二: 選擇對應的版本,點擊進入詳 ...
  • 拓展閱讀 資料庫設計工具-08-概覽 資料庫設計工具-08-powerdesigner 資料庫設計工具-09-mysql workbench 資料庫設計工具-10-dbdesign 資料庫設計工具-11-dbeaver 資料庫設計工具-12-pgmodeler 資料庫設計工具-13-erdplus ...
  • 初識STL STL,(Standard Template Library),即"標準模板庫",由惠普實驗室開發,STL中提供了非常多對信息學奧賽很有用的東西。 vector vetor是STL中的一個容器,可以看作一個不定長的數組,其基本形式為: vector<數據類型> 名字; 如: vector ...
  • 前言 最近自己做了個 Falsk 小項目,在部署上伺服器的時候,發現雖然不乏相關教程,但大多都是將自己項目代碼複製出來,不講核心邏輯,不太簡潔,於是將自己部署的經驗寫成內容分享出來。 uWSGI 簡介 uWSGI: 一種實現了多種協議(包括 uwsgi、http)並能提供伺服器搭建功能的 Pytho ...
  • 1 文本Embedding 將整個文本轉化為實數向量的技術。 Embedding優點是可將離散的詞語或句子轉化為連續的向量,就可用數學方法來處理詞語或句子,捕捉到文本的語義信息,文本和文本的關係信息。 ◉ 優質的Embedding通常會讓語義相似的文本在空間中彼此接近 ◉ 優質的Embedding相 ...