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
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...