成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Dotnet Core 技術之Dotnet 6.0 深度探索

開發(fā) 前端
引用微軟官方的說法:ConfigurationManager 是用來支持 ASP.Net Core 的新的 WebApplication 模型。這個模型主要的作用是在一些特定的場景下(后面我們會說到),用來簡化 ASP.NET Core 的啟動代碼。

[[442733]]

本文轉載自微信公眾號「老王Plus」,作者老王Plus的老王。轉載本文請聯(lián)系老王Plus公眾號。

Dotnet 6.0 大家都裝了沒?

我打算開個專題,系統(tǒng)地寫一寫 Dotnet 6.0 在各個方面的特性,以及全新的開發(fā)方式。也是因為最近討論 6.0 比較多,看到很多人的畏難情緒,所以打算寫寫相關的內(nèi)容。

了解了,就不怕了。

要寫的內(nèi)容很多,我會分幾篇來寫。

今天是第一篇:ConfigurationManager,配置管理器。

ConfigurationManager 是干什么用的?

引用微軟官方的說法:ConfigurationManager 是用來支持 ASP.Net Core 的新的 WebApplication 模型。這個模型主要的作用是在一些特定的場景下(后面我們會說到),用來簡化 ASP.NET Core 的啟動代碼。

當然,如果我們?nèi)タ?MSDN 的文檔,會發(fā)現(xiàn) ConfigurationManager 本身實現(xiàn)還是挺復雜的。好在,大多數(shù)情況下,這是一個半隱藏的東西,你可能都意識不到你已經(jīng)用到了它。

那它到底是干什么用的?

這得從 .Net 5.0 的 Configuration 說起。

.Net 5.0 里的 Configuration

Configuration 配置,從 3.1 到 5.0,增加了很多很多的配置類型,如果你去 MSDN 上看,有好幾大篇。

這里面,我們接觸最多的是兩個:

  • IConfigurationBuilder - 這個接口主要用來增加配置源,并在構建器上調用 Build() 來讀取每個配置源,并形成最終的配置
  • IConfigurationRoot - 這就是上面 Build() 完成后形成的配置,我們會從這里面讀配置值

在實際應用中,IConfigurationBuilder 通常被我們用做配置源列表的包裝器,最常用的是通過 AddJsonFile(),將配置源添加到源列表中。看到 AddJsonFile(),你是不是想到了什么?

簡單來說,IConfigurationBuilder 是這樣的:

  1. public interface IConfigurationBuilder 
  2.     IDictionary<string, object> Properties { get; } 
  3.     IList<IConfigurationSource> Sources { get; } 
  4.     IConfigurationBuilder Add(IConfigurationSource source); 
  5.     IConfigurationRoot Build(); 

而 IConfigurationRoot,里面放的是經(jīng)過合并的配置值。這個合并需要注意一下,多個配置源逐個加入時,相同名稱的項,后面的配置會覆蓋前面的項。

在 .Net 5.0 以前,IConfigurationBuilder 和 IConfigurationRoot 接口分別由 ConfigurationBuilder 和 ConfigurationRoot 實現(xiàn)。使用時通常是這么寫:

  1. var builder = new ConfigurationBuilder(); 
  2.  
  3. // 加入靜態(tài)值 
  4. builder.AddInMemoryCollection(new Dictionary<string, string> 
  5.     { "MyKey""MyValue" }, 
  6. }); 
  7.  
  8. // 加入文件 
  9. builder.AddJsonFile("appsettings.json"); 
  10.  
  11. IConfigurationRoot config = builder.Build(); 
  12.  
  13. string value = config["MyKey"]; // 取一個值 
  14. IConfigurationSection section = config.GetSection("SubSection"); // 取一個節(jié) 

這是在 Console 程序中。

在 ASP.NET Core 中,通常不需要這么顯式的 new 和 Build(),但事實上也是調用的這個接口。

在默認的 ConfigurationBuilder 實現(xiàn)中,調用 Build() 將遍歷所有的源,加載 Provider 程序,并將它們傳遞給一個新的ConfigurationRoot 實例:

  1. public IConfigurationRoot Build() 
  2.     var providers = new List<IConfigurationProvider>(); 
  3.     foreach (IConfigurationSource source in Sources) 
  4.     { 
  5.         IConfigurationProvider provider = source.Build(this); 
  6.         providers.Add(provider); 
  7.     } 
  8.     return new ConfigurationRoot(providers); 

然后,ConfigurationRoot 依次遍歷每個提供程序并加載配置值:

  1. public class ConfigurationRoot : IConfigurationRoot, IDisposable 
  2.     private readonly IList<IConfigurationProvider> _providers; 
  3.     private readonly IList<IDisposable> _changeTokenRegistrations; 
  4.  
  5.     public ConfigurationRoot(IList<IConfigurationProvider> providers) 
  6.     { 
  7.         _providers = providers; 
  8.         _changeTokenRegistrations = new List<IDisposable>(providers.Count); 
  9.  
  10.         foreach (IConfigurationProvider p in providers) 
  11.         { 
  12.             p.Load(); 
  13.             _changeTokenRegistrations.Add(ChangeToken.OnChange(() => p.GetReloadToken(), () => RaiseChanged())); 
  14.         } 
  15.     } 
  16.     // ...  

這種架構,會有個小問題。在團隊開發(fā)的時候,在沒有統(tǒng)一溝通的情況下,有可能會在多處調用 Build()。當然這也沒什么問題。只不過,正常來說這個沒有必要,畢竟這是在讀文件,會很慢。

不過,在 .Net 5.0 之前,都是這么做。

可喜的是,在 .Net 6.0 里,微軟也注意到這個問題,并引入了一個新的類型:ConfigurationManager。

.Net 6.0 里的 ConfigurationManager

ConfigurationManager 是一個 .Net 6.0 中新的配置類型。這個類型也同樣實現(xiàn)了兩個接口:IConfigurationBuilder 和 IConfigurationRoot。那么,通過這兩個接口的實現(xiàn),我們可以簡化上一節(jié)講到的 .Net 5.0 中的通用模式。

不過,還是有一點點區(qū)別。這里 IConfigurationBuilder 將源保存為 IList:

  1. public interface IConfigurationBuilder 
  2.     IList<IConfigurationSource> Sources { get; } 
  3.     // ... 

這樣做有一個好處,就是對于源 IList,就有了 Add() 和 Remove() 方法,我們可以在不知道 ConfigurationManager 的情況下增加和刪除配置提供程序。

  1. private class ConfigurationSources : IList<IConfigurationSource> 
  2.     private readonly List<IConfigurationSource> _sources = new(); 
  3.     private readonly ConfigurationManager _config; 
  4.  
  5.     public ConfigurationSources(ConfigurationManager config) 
  6.     { 
  7.         _config = config; 
  8.     } 
  9.  
  10.     public void Add(IConfigurationSource source) 
  11.     { 
  12.         _sources.Add(source); 
  13.         _config.AddSource(source); // 增加源 
  14.     } 
  15.  
  16.     public bool Remove(IConfigurationSource source) 
  17.     { 
  18.         var removed = _sources.Remove(source); // 刪除源 
  19.         _config.ReloadSources(); // 重新加載源 
  20.         return removed; 
  21.     } 
  22.  
  23.     // ...  

這樣做可以確保 ConfigurationManager 在改變源的 IList 時,能自動加載源的配置數(shù)據(jù)。

看一下 ConfigurationManager.AddSource 的定義:

  1. public class ConfigurationManager 
  2.     private void AddSource(IConfigurationSource source) 
  3.     { 
  4.         lock (_providerLock) 
  5.         { 
  6.             IConfigurationProvider provider = source.Build(this); 
  7.             _providers.Add(provider); 
  8.  
  9.             provider.Load(); 
  10.             _changeTokenRegistrations.Add(ChangeToken.OnChange(() => provider.GetReloadToken(), () => RaiseChanged())); 
  11.         } 
  12.  
  13.         RaiseChanged(); 
  14.     } 

這個方法會立即調用 IConfigurationSource 的 Build() 方法來創(chuàng)建 IConfigurationProvider,并加入到源列表中。

下面,這個方法就調用 IConfigurationProvider 的 Load() 方法,將數(shù)據(jù)加載到 Provider。

這個方法解決了一件事,就是當我們需要從不同的位置向 IConfigurationBuilder 加入各種源時,源只需要加載一次,而且只會加載一次。

上面的代碼是增加源。當我們需要 Remove() 源,或者干脆清除掉全部的源 Clear() 時,就需要調用 ReloadSource():

  1. private void ReloadSources() 
  2.     lock (_providerLock) 
  3.     { 
  4.         DisposeRegistrationsAndProvidersUnsynchronized(); 
  5.  
  6.         _changeTokenRegistrations.Clear(); 
  7.         _providers.Clear(); 
  8.  
  9.         foreach (var source in _sources) 
  10.         { 
  11.             _providers.Add(source.Build(this)); 
  12.         } 
  13.  
  14.         foreach (var p in _providers) 
  15.         { 
  16.             p.Load(); 
  17.             _changeTokenRegistrations.Add(ChangeToken.OnChange(() => p.GetReloadToken(), () => RaiseChanged())); 
  18.         } 
  19.     } 
  20.  
  21.     RaiseChanged(); 

當然,看懂上面的代碼,也就明白兩件事:

  • 增加源是代碼最小的,加到列表中就行了 ;
  • 刪除或更改源代碼會有點大,需要重新遍歷加載所有源。

如果需要對配置的源進行大量的操作,這樣的代價會比較大。不過,這種情況會很不常見。

總結一下

.Net 6.0 引入了一個新的 ConfigurationManager,用來優(yōu)化配置的構建。

ConfigurationManager 同樣實現(xiàn)了 ConfigurationBuilder 和 ConfigurationRoot。這算是個兼容性的設置,主要是為了支持 WebHostBuilder 和 HostBuilder 中對配置的調用。同時,也兼容了早期代碼中的調用方式。所以,代碼升級時,相關配置調用的部分,如果不想改代碼,是完全可以的。而如果想做點改動,就換成使用 ConfigurationManager,或者通過 WebApplicationBuilder 來加載(會自動調用 ConfigurationManager),應用程序會有更好的性能。

這算是一個小禮物,相信也是微軟權衡以后的結果。

 

責任編輯:武曉燕 來源: 老王Plus
相關推薦

2021-09-06 10:22:47

匿名對象編程

2024-05-21 10:23:02

反射技術.NET編程語言

2022-03-11 09:22:55

令牌Dotnet線程

2021-07-07 08:01:51

命令行Dotnet Core控制臺

2013-03-15 10:57:13

AJAXDotNet

2024-05-27 09:52:57

反射技術.NET動態(tài)庫

2021-01-13 08:11:45

項目結構類型

2021-01-20 08:16:06

異步Dotnet Core多路徑

2021-03-17 08:12:03

架構Dotnet洋蔥

2021-03-03 08:13:57

模式垃圾回收

2021-03-10 07:20:44

數(shù)據(jù)定位匹配

2021-06-02 08:07:59

LinuxService應用

2017-02-14 16:39:56

docker容器化主機

2011-04-20 17:15:21

并行計算

2011-04-21 09:13:14

并行計算

2021-02-03 08:12:23

函數(shù)委托Dotnet

2017-02-14 15:51:16

docker開發(fā)調試

2021-01-27 08:12:04

Dotnet函數(shù)數(shù)據(jù)

2022-11-22 08:01:34

dotNET 7API

2021-03-11 08:51:21

微軟漏洞dotnet
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美一区二区成人 | 欧美成年视频 | 成人免费视频网站在线看 | 8x国产精品视频一区二区 | 精品成人av | 精品久久久久久18免费网站 | 欧美精品一区二区在线观看 | 亚洲成人在线视频播放 | 中国大陆高清aⅴ毛片 | 色综合网站 | 国产精品久久777777 | 91视频观看 | 欧美影院 | 成人免费视频在线观看 | 久久1区| 久草高清视频 | 成人精品一区二区三区中文字幕 | 亚洲免费三级 | 午夜在线视频 | 91精品久久| 午夜男人免费视频 | 国产精品久久久久久久岛一牛影视 | 中文字幕在线观看av | 久久成人精品一区二区三区 | 国产亚洲精品久久19p | 91 久久 | 欧美综合在线观看 | 婷婷久久五月天 | 日韩91 | 日韩成人高清 | 荷兰欧美一级毛片 | 三级在线观看 | 一区二区三区四区免费视频 | 91免费在线| 亚洲自拍偷拍欧美 | 国产亚洲黄色片 | 日韩欧美在线视频观看 | 亚洲综合一区二区三区 | h视频免费在线观看 | 亚洲成人久久久 | 亚洲第一av|