IOptions、IOptionsMonitor以及IOptionsSnapshot

来源:https://www.cnblogs.com/wenhx/archive/2020/03/23/ioptions-ioptionsmonitor-and-ioptionssnapshot.html
-Advertisement-
Play Games

ASP.NET Core引入了Options模式,使用類來表示相關的設置組。簡單的來說,就是用強類型的類來表達配置項,這帶來了很多好處。 初學者會發現這個框架有3個主要的面向消費者的介面:IOptions ...


背景

ASP.NET Core引入了Options模式,使用類來表示相關的設置組。簡單的來說,就是用強類型的類來表達配置項,這帶來了很多好處。
初學者會發現這個框架有3個主要的面向消費者的介面:IOptions<TOptions>、IOptionsMonitor<TOptions>以及IOptionsSnapshot<TOptions>。
這三個介面初看起來很類似,所以很容易引起困惑,什麼場景下該用哪個介面呢?

示例

我們先從一小段代碼著手(TestOptions類只有一個字元串屬性Name,代碼略):

 1 class Program
 2 {
 3     static void Main(string[] args)
 4     {
 5         var builder = new ConfigurationBuilder();
 6         builder.AddJsonFile("appsettings.json", optional: true, reloadOnChange: true); //註意最後一個參數值,true表示配置文件更改時會重新載入。
 7         var configuration = builder.Build();
 8         var services = new ServiceCollection();
 9         services.AddOptions();
10         services.Configure<TestOptions>(configuration); //這裡通過配置文件綁定TestOptions
11         var provider = services.BuildServiceProvider();
12         Console.WriteLine("修改前:");
13         Print(provider);
14 
15         Change(provider); //使用代碼修改Options值。
16         Console.WriteLine("使用代碼修改後:");
17         Print(provider);
18         
19         Console.WriteLine("請修改配置文件。");
20         Console.ReadLine(); //等待手動修改appsettings.json配置文件。
21         Console.WriteLine("修改appsettings.json文件後:");
22         Print(provider);
23     }
24 
25     static void Print(IServiceProvider provider)
26     {
27         using(var scope = provider.CreateScope())
28         {
29             var sp = scope.ServiceProvider;
30             var options1 = sp.GetRequiredService<IOptions<TestOptions>>();
31             var options2 = sp.GetRequiredService<IOptionsMonitor<TestOptions>>();
32             var options3 = sp.GetRequiredService<IOptionsSnapshot<TestOptions>>();
33             Console.WriteLine("IOptions值: {0}", options1.Value.Name);
34             Console.WriteLine("IOptionsMonitor值: {0}", options2.CurrentValue.Name);
35             Console.WriteLine("IOptionsSnapshot值: {0}", options3.Value.Name);
36             Console.WriteLine();
37         }
38     }
39 
40     static void Change(IServiceProvider provider)
41     {
42         using(var scope = provider.CreateScope())
43         {
44             var sp = scope.ServiceProvider;
45             sp.GetRequiredService<IOptions<TestOptions>>().Value.Name = "IOptions Test 1";
46             sp.GetRequiredService<IOptionsMonitor<TestOptions>>().CurrentValue.Name = "IOptionsMonitor Test 1";
47             sp.GetRequiredService<IOptionsSnapshot<TestOptions>>().Value.Name = "IOptionsSnapshot Test 1";
48         }
49     }
50 }

appsettings.json文件:

{
    "Name": "Test 0"
}

上面的代碼,首先從appsettings.json文件讀取配置,然後向容器註冊依賴配置文件的TestOptions,接著分別列印IOptions<>,IOptionsMonitor<>和IOptionsSnapshot<>的值。

接著通過代碼來修改TestOptions的值,列印。
然後通過修改appsettings.json文件來修改TestOptions的值,列印。

註意,我們僅註冊了一次TestOptions,卻可以分別通過IOptions<>,IOptionsMonitor<>和IOptionsSnapshot<>介面來獲取TestOptions的值。

如果我們把appsettings.json文件中Name的值修改為Test 2,那麼上面這段代碼的輸出是這樣的:



分析

我們可以看到第一次通過代碼修改IOptions<>和IOptionsMonitor<>的值後,再次列印都被更新了,但是IOptionsSnapshot<>沒有,為什麼呢?
讓我們從Options框架的源代碼著手,理解為什麼會這樣。
當我們需要使用Options模式時,我們都會調用定義在OptionsServiceCollectionExtensions類上的擴展方法AddOptions(this IServiceCollection services)。

var services = new ServiceCollection();
services.AddOptions();

我們觀察AddOptions方法的實現:

public static IServiceCollection AddOptions(this IServiceCollection services)
{
    if (services == null)
    {
        throw new ArgumentNullException(nameof(services));
    }

    services.TryAdd(ServiceDescriptor.Singleton(typeof(IOptions<>), typeof(OptionsManager<>)));
    services.TryAdd(ServiceDescriptor.Scoped(typeof(IOptionsSnapshot<>), typeof(OptionsManager<>)));
    services.TryAdd(ServiceDescriptor.Singleton(typeof(IOptionsMonitor<>), typeof(OptionsMonitor<>)));
    services.TryAdd(ServiceDescriptor.Transient(typeof(IOptionsFactory<>), typeof(OptionsFactory<>)));
    services.TryAdd(ServiceDescriptor.Singleton(typeof(IOptionsMonitorCache<>), typeof(OptionsCache<>)));
    return services;
}

從上面的代碼我們可以得知,IOptions<>和IOptionsMonitor<>被註冊為單例服務,而IOptionsSnapshot<>被註冊為範圍服務。
由於IOptions<>和IOptionsMonitor<>都被註冊為單例服務,因此每次獲取的都是同一個實例,所以更改了以後的值是保留的。
而IOptionsSnapshot<>被註冊為範圍服務,所以每次創建新範圍時獲取的都是一個新的值,外部的更改只對當次有效,不會保留到下次(不能跨範圍,對於ASP.NET Core來說不能跨請求)。

我們繼續看第二次修改,第二次修改配置文件後IOptionsMonitor<>和IOptionsSnapshot<>的值更新了,而IOptions<>的值沒有更新。
IOptions<>好理解,它被註冊為單例服務,第一次訪問的時候生成實例並載入配置文件中的值,此後再也不會讀取配置文件,所以它的值不會更新。
IOptionsSnapshot<>被註冊為範圍服務,每次重新生成一個新的範圍時,它都會從配置文件中獲取值,因此它的值會更新。
但是,IOptionsMonitor<>呢,它被註冊為單例,為什麼也會更新呢?
讓我們回到AddOptions的源代碼,我們留意到IOptionsMonitor<>的實現是OptionsManager<>。
當我們打開OptionsManager的源代碼時,一切都很清楚了。
它的構造函數如下:

public OptionsMonitor(IOptionsFactory<TOptions> factory, IEnumerable<IOptionsChangeTokenSource<TOptions>> sources, IOptionsMonitorCache<TOptions> cache)
{
    _factory = factory;
    _sources = sources;
    _cache = cache;

    foreach (var source in _sources)
    {
        var registration = ChangeToken.OnChange(
                () => source.GetChangeToken(),
                (name) => InvokeChanged(name),
                source.Name);

        _registrations.Add(registration);
    }
}

原來OptionsMonitor的更新能力是從IOptionsChangeTokenSource<TOptions>而來,但是這個介面的實例又是誰呢?
我們回到最開始的代碼的第10行:

services.Configure<TestOptions>(configuration);

這是一個定義在Microsoft.Extensions.Options.ConfigurationExtensions.dll的擴展方法,最後實際調用的是它的一個重載方法,代碼如下:

 1 public static IServiceCollection Configure<TOptions>(this IServiceCollection services, string name, IConfiguration config, Action<BinderOptions> configureBinder)
 2     where TOptions : class
 3 {
 4     if (services == null)
 5     {
 6         throw new ArgumentNullException(nameof(services));
 7     }
 8 
 9     if (config == null)
10     {
11         throw new ArgumentNullException(nameof(config));
12     }
13 
14     services.AddOptions();
15     services.AddSingleton<IOptionsChangeTokenSource<TOptions>>(new ConfigurationChangeTokenSource<TOptions>(name, config));
16     return services.AddSingleton<IConfigureOptions<TOptions>>(new NamedConfigureFromConfigurationOptions<TOptions>(name, config, configureBinder));
17 }

秘密就在上面的第15行,ConfigurationChangeTokenSource,它引用了代表配置文件的對象config,所以配置文件更新,IOptionsMonitor就會跟著更新。

結論

IOptions<>是單例,因此一旦生成了,除非通過代碼的方式更改,它的值是不會更新的。
IOptionsMonitor<>也是單例,但是它通過IOptionsChangeTokenSource<> 能夠和配置文件一起更新,也能通過代碼的方式更改值。
IOptionsSnapshot<>是範圍,所以在配置文件更新的下一次訪問,它的值會更新,但是它不能跨範圍通過代碼的方式更改值,只能在當前範圍(請求)內有效。

官方文檔是這樣介紹的:
IOptionsMonitor<TOptions>用於檢索選項和管理TOptions實例的選項通知,它支持下麵的場景:

  • 實例更新通知。
  • 命名實例。
  • 重新載入配置。
  • 選擇性的讓實例失效。

IOptionsSnapshot<TOptions>在需要對每個請求重新計算選項的場景中非常有用。
IOptions<TOptions>可以用來支持Options模式,但是它不支持前面兩者所支持的場景,如果你不需要支持上面的場景,你可以繼續使用IOptions<TOptions>。

所以你應該根據你的實際使用場景來選擇到底是用這三者中的哪一個。
一般來說,如果你依賴配置文件,那麼首先考慮IOptionsMonitor<>,如果不合適接著考慮IOptionsSnapshot<>,最後考慮IOptions<>。
有一點需要註意,在ASP.NET Core應用中IOptionsMonitor可能會導致同一個請求中選項的值不一致——當你正在修改配置文件的時候——這可能會引發一些奇怪的bug。
如果這個對你很重要,請使用IOptionsSnapshot,它可以保證同一個請求中的一致性,但是它可能會帶來輕微的性能上的損失。
如果你是在app啟動的時候自己構造Options(比如在Startup類中):

services.Configure<TestOptions>(opt => opt.Name = "Test 0");

IOptions<>最簡單,也許是一個不錯的選擇,Configure擴展方法還有其他重載可以滿足你的更多需求。


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

-Advertisement-
Play Games
更多相關文章
  • 在Java 7發行版中,oracle在異常處理機制上也做了一些不錯的更改。這些主要是 改進的catch塊 和 多餘的throws子句 。讓我們看看他們是如何改變的。 1.改進了Java 7中的catch塊 在此功能中,現在您可以 在單個catch塊中捕獲多個異常 。在Java 7之前,您只能在每個c ...
  • 入門訓練3 圓的面積 問題描述 給定圓的半徑r,求圓的面積。 輸入格式 輸入包含一個整數r,表示圓的半徑。 輸出格式 輸出一行,包含一個實數,四捨五入保留小數點後7位,表示圓的面積。 說明:在本題中,輸入是一個整數,但是輸出是一個實數。 對於實數輸出的問題,請一定看清楚實數輸出的要求,比如本題中要求 ...
  • 第一章 認識Java8以及函數式編程 儘管距離Java8發佈已經過去7、8年的時間,但時至今日仍然有許多公司、項目停留在Java7甚至更早的版本。即使已經開始使用Java8的項目,大多數程式員也仍然採用“傳統”的編碼方式。 即使是在Java7就已經有了處理異常的新方式—— ,但大多數程式員也仍然採用 ...
  • 前言 2019 年底開始蔓延的新型肺炎疫情牽動人心,作為個體,我們力所能及的就是儘量待在家中少出門。 看到一些朋友叫設計同學幫忙給自己的頭像戴上口罩,作為技術人,心想一定還有更多人有這樣的訴求,不如開發一個簡單的程式來實現這個需求,也算是幫助設計姐姐減少工作量。 於是花了些時間,寫了一個叫做 fac ...
  • 企業應用中,涉及到修改狀態的場景太多了。比如,企業入網後,要審核資質。個人領取任務後,企業管理員要審核領取人。 應用管理系統中,通常是下圖這樣,在列表後有操作按鈕來修改數據記錄的狀態。 點擊“通過”/“拒絕”操作,要修改數據記錄的status欄位。服務端程式邏輯怎麼實現呢? 先定義服務端api介面: ...
  • @2020.3.23 課後練習——裝飾器 一:編寫函數,(函數執行的時間用time.sleep(n)模擬) 二:編寫裝飾器,為函數加上統計時間的功能 三:編寫裝飾器,為函數加上認證的功能 四:編寫裝飾器,為多個函數加上認證的功能(用戶的賬號密碼來源於文件),要求登錄成功一次,後續的函數都無需再輸入用 ...
  • 用於對類文件進行分類管理,給類文件提供多層命名空間,類名的全稱是:包名.類名,包也是一種封裝形式。 javac -d 目錄 *.java 通過package被訪問的類和類中成員要public修飾。 不同包中的子類還可以直接訪問父類中被protected許可權修飾的成員。 包與包之間可以使用的許可權有兩種 ...
  • 背景介紹: 項目是微服務的,使用docker容器,使用jenkins部署。測試環境有個公共服務一直以來都能正常發佈,突然有一天不行了,經常發佈失敗,然後多發佈幾次就好了。 報錯如下: 是棧溢出了,一般是新代碼有死迴圈會出現。但是本地啟動沒問題並且環境上多發幾次也能成功,說明沒有死迴圈,肯定是其他原因 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...