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.講故事 在分析的眾多dump中,經常會遇到各種奇葩的問題,僅通過dump這種快照形式還是有很多問題搞不定,而通過 perfview 這種粒度又太粗,很難找到問題之所在,真的很頭疼,比如本篇的 短命線程 問題,參考圖如下: 我們在 t2 時刻抓取的dump對查看 短命線程 毫無幫助,我根 ...
  • 在日常後端Api開發中,我們跟前端的溝通中,通常需要協商好入參的數據類型,和參數是通過什麼方式存在於請求中的,是表單(form)、請求體(body)、地址欄參數(query)、還是說通過請求頭(header)。 當協商好後,我們的介面又需要怎麼去接收這些數據呢?很多小伙伴可能上手就是直接寫一個實體, ...
  • 許多情況下我們需要用到攝像頭獲取圖像,進而處理圖像,這篇博文介紹利用pyqt5、OpenCV實現用電腦上連接的攝像頭拍照並保存照片。為了使用和後續開發方便,這裡利用pyqt5設計了個相機界面,後面將介紹如何實現,要點包括界面設計、邏輯實現及完整代碼。 ...
  • 思路分析 註冊頁面需要對用戶提交的數據進行校驗,並且需要對用戶輸入錯誤的地方進行提示! 所有我們需要使用forms組件搭建註冊頁面! 平時我們書寫form是組件的時候是在views.py裡面書寫的, 但是為了接耦合,我們需要將forms組件都單獨寫在一個地方,需要用的時候導入就行! 例如,在項目文件 ...
  • 思路分析 登錄頁面,我們還是採用ajax的方式提交用戶數據 唯一需要學習的是如何製作圖片驗證碼! 具體的登錄頁面效果圖如下: 如何製作圖片驗證碼 推導步驟1:在img標簽的src屬性里放上驗證碼的請求路徑 補充1.img的src屬性: 1.圖片路徑 2.url 3.圖片的二進位數據 補充2:字體樣式 ...
  • 哈嘍,兄弟們! 最近有許多小伙伴都在吐槽打工好難。 每天都是執行許多重覆的任務 例如閱讀新聞、發郵件、查看天氣、打開書簽、清理文件夾等等, 使用自動化腳本,就無需手動一次又一次地完成這些任務, 非常方便啊有木有?! 而在某種程度上,Python 就是自動化的代名詞。 今天就來和大家一起學習一下, 用 ...
  • 作者:IT王小二 博客:https://itwxe.com 前面小二介紹過使用Typora+PicGo+LskyPro打造舒適寫作環境,那時候需要使用水印功能,但是小二在升級LskyPro2.x版本發現有很多不如人意的東西,遂棄用LskyPro使用MinIO結合代碼實現自己需要的圖床功能,也適合以後 ...
  • OpenAI Gym是一款用於研發和比較強化學習演算法的工具包,本文主要介紹Gym模擬環境的功能和工具包的使用方法,並詳細介紹其中的經典控制問題中的倒立擺(CartPole-v0/1)問題。最後針對倒立擺問題如何建立控制模型並採用爬山演算法優化進行了介紹,並給出了相應的完整python代碼示例和解釋。要... ...
  • python爬蟲瀏覽器偽裝 #導入urllib.request模塊 import urllib.request #設置請求頭 headers=("User-Agent","Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, l ...
  • 前端代碼搭建 主要利用的是bootstrap3中js插件里的模態框版塊 <li><a href="" data-toggle="modal" data-target=".bs-example-modal-lg">修改密碼</a></li> <div class="modal fade bs-exam ...