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
  • 1. 說明 /* Performs operations on System.String instances that contain file or directory path information. These operations are performed in a cross-pla ...
  • 視頻地址:【WebApi+Vue3從0到1搭建《許可權管理系統》系列視頻:搭建JWT系統鑒權-嗶哩嗶哩】 https://b23.tv/R6cOcDO qq群:801913255 一、在appsettings.json中設置鑒權屬性 /*jwt鑒權*/ "JwtSetting": { "Issuer" ...
  • 引言 集成測試可在包含應用支持基礎結構(如資料庫、文件系統和網路)的級別上確保應用組件功能正常。 ASP.NET Core 通過將單元測試框架與測試 Web 主機和記憶體中測試伺服器結合使用來支持集成測試。 簡介 集成測試與單元測試相比,能夠在更廣泛的級別上評估應用的組件,確認多個組件一起工作以生成預 ...
  • 在.NET Emit編程中,我們探討了運算操作指令的重要性和應用。這些指令包括各種數學運算、位操作和比較操作,能夠在動態生成的代碼中實現對數據的處理和操作。通過這些指令,開發人員可以靈活地進行算術運算、邏輯運算和比較操作,從而實現各種複雜的演算法和邏輯......本篇之後,將進入第七部分:實戰項目 ...
  • 前言 多表頭表格是一個常見的業務需求,然而WPF中卻沒有預設實現這個功能,得益於WPF強大的控制項模板設計,我們可以通過修改控制項模板的方式自己實現它。 一、需求分析 下圖為一個典型的統計表格,統計1-12月的數據。 此時我們有一個需求,需要將月份按季度劃分,以便能夠直觀地看到季度統計數據,以下為該需求 ...
  • 如何將 ASP.NET Core MVC 項目的視圖分離到另一個項目 在當下這個年代 SPA 已是主流,人們早已忘記了 MVC 以及 Razor 的故事。但是在某些場景下 SSR 還是有意想不到效果。比如某些靜態頁面,比如追求首屏載入速度的時候。最近在項目中回歸傳統效果還是不錯。 有的時候我們希望將 ...
  • System.AggregateException: 發生一個或多個錯誤。 > Microsoft.WebTools.Shared.Exceptions.WebToolsException: 生成失敗。檢查輸出視窗瞭解更多詳細信息。 內部異常堆棧跟蹤的結尾 > (內部異常 #0) Microsoft ...
  • 引言 在上一章節我們實戰了在Asp.Net Core中的項目實戰,這一章節講解一下如何測試Asp.Net Core的中間件。 TestServer 還記得我們在集成測試中提供的TestServer嗎? TestServer 是由 Microsoft.AspNetCore.TestHost 包提供的。 ...
  • 在發現結果為真的WHEN子句時,CASE表達式的真假值判斷會終止,剩餘的WHEN子句會被忽略: CASE WHEN col_1 IN ('a', 'b') THEN '第一' WHEN col_1 IN ('a') THEN '第二' ELSE '其他' END 註意: 統一各分支返回的數據類型. ...
  • 在C#編程世界中,語法的精妙之處往往體現在那些看似微小卻極具影響力的符號與結構之中。其中,“_ =” 這一組合突然出現還真不知道什麼意思。本文將深入剖析“_ =” 的含義、工作原理及其在實際編程中的廣泛應用,揭示其作為C#語法奇兵的重要角色。 一、下劃線 _:神秘的棄元符號 下劃線 _ 在C#中並非 ...