ASP.NET Core - 選項系統之選項使用

来源:https://www.cnblogs.com/wewant/archive/2023/03/23/17111653.html
-Advertisement-
Play Games

上一篇 ASP.NET Core - 選項系統之選項配置 中提到 IOptions、IOptionsMonitor 和 IOptionsSnapshot 三個介面,通過這三個介面都可以從依賴註入容器中解析出已經配置的選項類,在我們通過 Configure 方法配置選項時,這三個介面會被同時註冊,但三 ...


上一篇 ASP.NET Core - 選項系統之選項配置 中提到 IOptions、IOptionsMonitor 和 IOptionsSnapshot 三個介面,通過這三個介面都可以從依賴註入容器中解析出已經配置的選項類,在我們通過 Configure 方法配置選項時,這三個介面會被同時註冊,但三個介面是有區別的,適用場景也有所不同。

1. IOptions

  • IOptions 對象的生命周期是 Singleton (單例),它可以在任意地方進行註入使用
  • 該介面對象在第一次使用的時候被實例化,並且選項類中的內容會一直保持不變,前面也提過選項類內容可以在配置來源修改之後更新,但是通過 IOption 解析的選項類不會隨著更新而改變
  • IOptions 介面不支持命名選項模式,它是沒有 get 方法的,也並不會預設讀取第一個,它只能讀取 String.Empty 預設命名的選項,如果沒有配置預設選項的話,雖然也能解析出 Options 選項類對象,但是對象的屬性都是相應類型的預設值(引用類型是 null,值類型是 0,其他的也都是相應類型的預設值)
public class OptionController : ControllerBase
{
	private readonly BlogOptions _blogOptions;
	public OptionController(IOptions<BlogOptions> options)
	{
		// 通過 IOptions<TOptions> 介面的 Value 屬性讀取選項類
		// 選項類始終是程式啟動時載入的值,不會改變
		_blogOptions = options.Value;
	}
}

2. IOptionsMonitor

  • IOptionsMonitor 對象的生命周期是 Scoped(作用域),Scoped 生命周期的特點是不能註入到 Singleton 服務中
  • 在作用域中(最常見的一次Http請求),創建 IOptionsSnapshot 對象實例時,會從配置中讀取最新選項值作為快照,併在當前作用域中始終使用該快照。也就是說一次請求中選項類內容保持不變,但是不同請求中可能因為配置來源的修改而不同
  • IOptionsMonitor 支持命名選項
public class OptionController : ControllerBase
{
	private readonly BlogOptions _blogOptions;
	public OptionController(IOptionsSnapshot<BlogOptions> optionsSnapshot)
	{
		// IOptionsSnapshot<TOptions> 可以通過 Value 屬性讀取預設的命名的選項類, Options 對象實例創建時讀取的配置快照
		_blogOptions = optionsSnapshot.Value;
		// 也可以通過 Get 方法獲取某一個命名選項,沒有指定命名時,預設命名為 string.Empty
		//_blogOptions = optionsSnapshot.Get(string.Empty);
	}
}

3. IOptionsSnapshot

  • IOptionsSnapshot 對象的生命周期也是 Singleton (單例)
  • 通過 IOptionsSnapshot 介面註入的對象每次讀取選項值時,都是從配置中讀取最新選項值,能夠實時獲取配置來源的更改
  • 該介面支持命名選項模式
  • 除了可以查看 TOptions 的值,還可以監控 TOptions 配置的更改,支持重新載入配置(CurrentValue),並當配置發生更改時,進行通知(OnChange),支持緩存與緩存失效 (IOptionsMonitorCache),每次調用實例的 CurrentValue 時,會先檢查緩存(IOptionsMonitorCache)是否有值,如果有值,則直接用,如果沒有,則從配置中讀取最新選項值,並記入緩存。當配置發生更改時,會將緩存清空。
public class OptionController : ControllerBase
{
	private readonly BlogOptions _blogOptions;
	public OptionController(IOptionsMonitor<BlogOptions> optionsMonitor)
	{
		// IOptionsMonitor<TOptions> 介面沒有 Value 屬性,通過 CurrentValue 獲取選項類對象,
		// 每次調用 CurrentValue都會實時讀取配置源,始終是最新配置的值
		_blogOptions = optionsMonitor.CurrentValue;
		// 該介面也支持通過 Get 方法獲取命名選項
		_blogOptions = optionsMonitor.Get(string.Empty);
		// 可以通過 OnChange 註冊事件,當配置被載入時會觸發事件
		optionsMonitor.OnChange(OnOptionsChange);
	}

	[HttpGet]
	public Task<BlogOptions> Get()
	{
		return Task.FromResult(_blogOptions);
	}

	private void OnOptionsChange(BlogOptions options)
	{
		Console.WriteLine(JsonSerializer.Serialize(options));
	}
}

啟動應用,調用一次 Get 介面,在 Api 控制器構造函數中註冊了配置載入觸發事件,之後修改 appsettings.json 配置文件中選項類對於的配置節點內容,可以看到事件觸發,控制臺中輸出了改變之後的選項類內容。

image

4. 三個介面的選項讀取機制演示

三個介面解析的選項類的差別,可以通過以下測試清楚得看出:

配置文件中初始選項節點如下:

"Blog": {
    "Title": "ASP.NET Core Options11",
    "Content": "This is a blog about Options System in ASP.NET Core Framework.",
    "CreateTime": "2022-12-06"
}

這裡為了方便看出 Scoped 生命周期 IOptionSnapeshoot 介面的變化,所有通過 Web Api 介面來測試

public class OptionController : ControllerBase
{
	private readonly IOptions<BlogOptions> _blogOptions;
	private readonly IOptionsSnapshot<BlogOptions> _blogSnapshotOptions;
	private readonly IOptionsMonitor<BlogOptions> _blogMonitorOptions;
	public OptionController(
		 IOptions<BlogOptions> options,
		IOptionsSnapshot<BlogOptions> optionsSnapshot,
		IOptionsMonitor<BlogOptions> optionsMonitor
		)
	{
// 註意這裡不能再把選項類對象先讀取出來,否則選項類對象也不會再改變了
		_blogOptions = options;
		_blogSnapshotOptions = optionsSnapshot;
		_blogMonitorOptions = optionsMonitor;
	}

	[HttpGet]
	public Task Get()
	{
		Console.WriteLine("第一次讀取配置:");
		Console.WriteLine("IOptions<TOptions>:" + JsonSerializer.Serialize(_blogOptions.Value));
		Console.WriteLine("IOptionsSnapshot<TOptions>:" + JsonSerializer.Serialize(_blogSnapshotOptions.Value));
		Console.WriteLine("IOptionsMonitor<TOptions>:" + JsonSerializer.Serialize(_blogMonitorOptions.CurrentValue));

		Console.WriteLine("請修改配置文件!");
		Console.ReadKey();

		Console.WriteLine("第二次讀取配置:");
		Console.WriteLine("IOptions<TOptions>:" + JsonSerializer.Serialize(_blogOptions.Value));
		Console.WriteLine("IOptionsSnapshot<TOptions>:" + JsonSerializer.Serialize(_blogSnapshotOptions.Value));
		Console.WriteLine("IOptionsMonitor<TOptions>:" + JsonSerializer.Serialize(_blogMonitorOptions.CurrentValue));

		return Task.CompletedTask;
	}
}

之後啟動應用調用 Get 介面,併在過程中將配置文件內容修改為以下:

"Blog": {
    "Title": "ASP.NET Core Options222",
    "Content": "This is a blog about Options System in ASP.NET Core Framework.",
    "CreateTime": "2022-12-06"
}

可以看到控制台的輸出中,第二次讀取配置的時候,IOptionsMonitor 介面讀取到的內容已經改變

image

之後不要關閉應用,再調一次 Get 介面,併在過程中再次修改配置如下:

"Blog": {
    "Title": "ASP.NET Core Options333",
    "Content": "This is a blog about Options System in ASP.NET Core Framework.",
    "CreateTime": "2022-12-06"
}

這一次的 Get 請求的輸出結果如下:
image

可以看到 IOptionsMonitor 介面每次都能獲取到配置文件的實時值,IOptionsSnapshot 介面相較於第一次調用 Get 介面的時候已經改變,獲取到了之前修改的值,但是之後的修改它又獲取不到了,因為它是 Scoped 生命周期,在一次請求內是保持一致的,而 IOptions 介面獲取到的選項類對象是一致不變的。



參考文章:
ASP.NET Core 中的選項模式 | Microsoft Learn
選項模式 - .NET | Microsoft Learn
面向 .NET 庫創建者的選項模式指南 - .NET | Microsoft Learn
理解ASP.NET Core - 選項(Options)



ASP.NET Core 系列:

目錄:ASP.NET Core 系列總結
上一篇:ASP.NET Core - 選項系統之選項配置


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 記憶體重疊是指在記憶體中存在兩個或多個區域,它們的地址範圍有交叉部分。在 C++ 中,記憶體重疊可能會導致程式出現不可預期的行為,因此我們需要瞭解它的原因和如何避免。 記憶體重疊的原因 記憶體重疊的主要原因是指針的使用。當我們使用指針訪問記憶體時,如果指針指向的記憶體區域與另一個區域有交叉部分,就會產生記憶體重疊。 ...
  • Mybatis高級特性能夠幫助我們更加靈活地操作資料庫,包括動態SQL、緩存機制、插件機制、自定義類型轉換等。學習這些特性可以讓我們更好地利用Mybatis,提高數據操作的效率和質量。 未來的道路由自己抉擇,事業的高度由自己決定。 動態SQL 動態SQL中,Mybatis提供了多種標簽來幫助我們構建 ...
  • 一、問題引入 在學習棧的過程中,教材有一個案例:利用棧結果解析括弧的匹配問題。括弧問題:[({}{})],說明 [] 、() 、{} 稱為一對。 號碼位置對應的括弧之間進行匹配,結果:0-7、 1-6、 2-3、 4-5 二、過程記錄 💡 基於順序棧實現 利用棧的特性:先進後出 ,對括弧進行匹配輸 ...
  • 本系列將和大家分享Redis分散式緩存,本章主要簡單介紹下Redis中的布隆過濾器(Bloom Filter),以及如何破解ServiceStack和如何解決緩存雪崩、緩存穿透、緩存擊穿、緩存預熱問題。 ...
  • 一、Show與ShowDialog眾所周知在c#中有兩種顯示視窗的方式:模態顯示(showdialog)與非模態顯示(show),模態顯示會阻塞調用視窗的所有消息響應,在調用ShowDialog方法後,直到關閉對話框後,才執行此方法後面的代碼 ,期間用戶是無法對該視窗外的界面進行ui交互的;非模態顯 ...
  • 背景: 如何在ASP.Net Core的生產環境中保護swagger ui,也就是index.html頁面。其實swagger是自帶禁用的功能的,只需要設置開關即可。但是有一些場景,是需要把這些介面進行開放或者導出成文檔供第三方進行調用,這個時候卻又不想讓所有人訪問。本文介紹一種許可權控制訪問的方式, ...
  • 一:背景 1. 講故事 最近收到了兩起程式崩潰的dump,查了下都是經典的 double free 造成的,蠻有意思,這裡就抽一篇出來分享一下經驗供後面的學習者避坑吧。 二:WinDbg 分析 1. 崩潰點在哪裡 windbg 帶了一個自動化分析命令 !analyze -v 可以幫助我們找到崩潰時的 ...
  • .net core 自定義授權策略提供程式進行許可權驗證 在這之前先瞭解一下鑒權和授權的概念; 鑒權 鑒權可以說是身份驗證,身份驗證是確定用戶身份的過程; 在ASP.NET Core 中身份驗證是由身份驗證服務IAuthenticationService負責的,它被身份驗證中間件使用, 身份驗證服務會 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...