国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - ASP.NET教程 - .NET Core 3.0之創建基于Consul的Configuration擴展組件

.NET Core 3.0之創建基于Consul的Configuration擴展組件

2020-06-15 14:11laozhang ASP.NET教程

在本文里小編給大家分享了關于.NET Core 3.0之創建基于Consul的Configuration擴展組件相關知識點,需要的朋友們學習下。

經過前面三篇關于.NET Core Configuration的文章之后,本篇文章主要討論如何擴展一個Configuration組件出來。

了解了Configuration的源碼后,再去擴展一個組件就會比較簡單,接下來我們將在.NET Core 3.0-preview5的基礎上創建一個基于Consul的配置組件。

相信大家對Consul已經比較了解了,很多項目都會使用Consul作為配置中心,此處也不做其他闡述了,主要是講一下,創建Consul配置擴展的一些思路。使用Consul配置功能時,我們可以將信息轉成JSON格式后再存儲,那么我們在讀取的時候,在體驗上就像是從讀取JSON文件中讀取一樣。

開發前的準備初始化Consul

假設你已經安裝并啟動了Consul,我們打開Key/Value功能界面,創建兩組配置選項出來,分別是commonservice和userservice,如下圖所示

.NET Core 3.0之創建基于Consul的Configuration擴展組件

配置值采用JSON格式

.NET Core 3.0之創建基于Consul的Configuration擴展組件

實現思路

我們知道在Configuration整個的設計框架里,比較重要的類ConfigurationRoot,內部又有一個IConfigurationProvider集合屬性,也就是說我們追加IConfigurationProvider實例最終也會被放到到該集合中,如下圖所示

.NET Core 3.0之創建基于Consul的Configuration擴展組件

該項目中,我使用到了一個已經封裝好的Consul(V0.7.2.6)類庫,同時基于.NET Core關于Configuration的設計風格,做如下的框架設計

.NET Core 3.0之創建基于Consul的Configuration擴展組件

考慮到我會在該組件內部創建ConsulClient實例,所以對ConsulClient構造函數的一部分參數做了抽象提取,并添加到了IConsulConfigurationSource中,以增強該組件的靈活性。

之前說過,Consul中的配置信息是以JSON格式存儲的,所以此處使用到了Microsoft.Extensions.Configuration.Json.JsonConfigurationFileParser,用以將JSON格式的信息轉換為Configuration的通用格式Key/Value。

核心代碼 IConsulConfigurationSource

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
/// <summary>
 /// ConsulConfigurationSource
 /// </summary>
public interface IConsulConfigurationSource : IConfigurationSource
 {
  /// <summary>
  /// CancellationToken
  /// </summary>
  CancellationToken CancellationToken { get; }
 
  /// <summary>
  /// Consul構造函數實例,可自定義傳入
  /// </summary>
  Action<ConsulClientConfiguration> ConsulClientConfiguration { get; set; }
 
  /// <summary>
  /// Consul構造函數實例,可自定義傳入
  /// </summary>
  Action<HttpClient> ConsulHttpClient { get; set; }
 
  /// <summary>
  /// Consul構造函數實例,可自定義傳入
  /// </summary>
  Action<HttpClientHandler> ConsulHttpClientHandler { get; set; }
 
  /// <summary>
  /// 服務名稱
  /// </summary>
  string ServiceKey { get; }
 
  /// <summary>
  /// 可選項
  /// </summary>
  bool Optional { get; set; }
 
  /// <summary>
  /// Consul查詢選項
  /// </summary>
  QueryOptions QueryOptions { get; set; }
 
  /// <summary>
  /// 重新加載延遲時間,單位是毫秒
  /// </summary>
  int ReloadDelay { get; set; }
 
  /// <summary>
  /// 是否在配置改變的時候重新加載
  /// </summary>
  bool ReloadOnChange { get; set; }
 }

 

ConsulConfigurationSource

該類提供了一個構造函數,用于接收ServiceKey和CancellationToken實例

?
1
2
3
4
5
6
7
8
9
10
public ConsulConfigurationSource(string serviceKey, CancellationToken cancellationToken)
 {
 if (string.IsNullOrWhiteSpace(serviceKey))
 {
  throw new ArgumentNullException(nameof(serviceKey));
 }
 
 this.ServiceKey = serviceKey;
 this.CancellationToken = cancellationToken;
}

 

其build()方法也比較簡單,主要是初始化ConsulConfigurationParser實例

?
1
2
3
4
5
6
public IConfigurationProvider Build(IConfigurationBuilder builder)
{
 ConsulConfigurationParser consulParser = new ConsulConfigurationParser(this);
 
 return new ConsulConfigurationProvider(this, consulParser);
}

 

ConsulConfigurationParser

該類比較復雜,主要實現Consul配置的獲取、監控以及容錯處理,公共方法源碼如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
/// <summary>
/// 獲取并轉換Consul配置信息
/// </summary>
/// <param name="reloading"></param>
/// <param name="source"></param>
/// <returns></returns>
public async Task<IDictionary<string, string>> GetConfig(bool reloading, IConsulConfigurationSource source)
{
 try
 {
  QueryResult<KVPair> kvPair = await this.GetKvPairs(source.ServiceKey, source.QueryOptions, source.CancellationToken).ConfigureAwait(false);
  if ((kvPair?.Response == null) && !source.Optional)
  {
   if (!reloading)
   {
    throw new FormatException(Resources.Error_InvalidService(source.ServiceKey));
   }
 
   return new Dictionary<string, string>();
  }
 
  if (kvPair?.Response == null)
  {
   throw new FormatException(Resources.Error_ValueNotExist(source.ServiceKey));
  }
 
  this.UpdateLastIndex(kvPair);
 
  return JsonConfigurationFileParser.Parse(source.ServiceKey, new MemoryStream(kvPair.Response.Value));
 }
 catch (Exception exception)
 {
  throw exception;
 }
}
 
/// <summary>
/// Consul配置信息監控
/// </summary>
/// <param name="key"></param>
/// <param name="cancellationToken"></param>
/// <returns></returns>
public IChangeToken Watch(string key, CancellationToken cancellationToken)
{
 Task.Run(() => this.RefreshForChanges(key, cancellationToken), cancellationToken);
 
 return this.reloadToken;
}

 

另外,關于Consul的監控主要利用了QueryResult.LastIndex屬性,該類緩存了該屬性的值,并與實獲取的值進行比較,以判斷是否需要重新加載內存中的緩存配置

ConsulConfigurationProvider

該類除了實現Load方法外,還會根據ReloadOnChange屬性,在構造函數中注冊OnChange事件,用于重新加載配置信息,源碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
public sealed class ConsulConfigurationProvider : ConfigurationProvider
 {
  private readonly ConsulConfigurationParser configurationParser;
  private readonly IConsulConfigurationSource source;
 
  public ConsulConfigurationProvider(IConsulConfigurationSource source, ConsulConfigurationParser configurationParser)
  {
   this.configurationParser = configurationParser;
   this.source = source;
 
   if (source.ReloadOnChange)
   {
    ChangeToken.OnChange(
     () => this.configurationParser.Watch(this.source.ServiceKey, this.source.CancellationToken),
     async () =>
     {
      await this.configurationParser.GetConfig(true, source).ConfigureAwait(false);
 
      Thread.Sleep(source.ReloadDelay);
 
      this.OnReload();
     });
   }
  }
 
  public override void Load()
  {
   try
   {
    this.Data = this.configurationParser.GetConfig(false, this.source).ConfigureAwait(false).GetAwaiter().GetResult();
   }
   catch (AggregateException aggregateException)
   {
    throw aggregateException.InnerException;
   }
  }
 }
 

調用及運行結果

此處調用在Program中實現

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public class Program
{
 public static void Main(string[] args)
 {
  CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
 
  WebHost.CreateDefaultBuilder(args).ConfigureAppConfiguration(
   (hostingContext, builder) =>
   {
    builder.AddConsul("userservice", cancellationTokenSource.Token, source =>
    {
     source.ConsulClientConfiguration = cco => cco.Address = new Uri("http://localhost:8500");
     source.Optional = true;
     source.ReloadOnChange = true;
     source.ReloadDelay = 300;
     source.QueryOptions = new QueryOptions
     {
      WaitIndex = 0
     };
    });
 
    builder.AddConsul("commonservice", cancellationTokenSource.Token, source =>
    {
     source.ConsulClientConfiguration = cco => cco.Address = new Uri("http://localhost:8500");
     source.Optional = true;
     source.ReloadOnChange = true;
     source.ReloadDelay = 300;
     source.QueryOptions = new QueryOptions
     {
      WaitIndex = 0
     };
    });
   }).UseStartup<Startup>().Build().Run();
 }
}

以上就是本次介紹的全部知識點內容,感謝大家對服務器之家的支持。

 

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: av中文字幕在线观看 | 欧美黄视频在线观看 | 国产精品久久久久久一区 | 在线视频 中文字幕 | 亚洲国产91 | 在线永久免费观看日韩a | 欧美日韩成人在线视频 | 欧美性猛交一区二区三区精品 | 欧美精品久久久久久久久老牛影院 | 精品伊人 | 欧美一区二区三区 | 成年人在线观看 | 欧美在线亚洲 | 成人免费视频在线观看 | 午夜精品久久久久久久久 | 亚洲一区二区三区四区的 | 亚洲日韩中文字幕一区 | 国产精品一区二区视频 | 国产精品无码永久免费888 | 成人午夜视频在线播放 | 国产一区二区三区在线视频 | 欧美一级在线观看 | 我要看a级毛片 | 狠狠搞狠狠干 | 久操色| 国产成人精品一区二区三区福利 | 91综合国产 | 欧美日韩激情 | 在线免费观看av电影 | 亚洲一区二区三区精品动漫 | 国产中文字幕一区 | 男人的天堂在线视频 | 精品九九 | 精品无码久久久久久国产 | 久久精品久久久 | 精品久久久一 | 午夜网 | www国产网站 | 激情五月婷婷在线 | 日韩一区中文字幕 | 91精品啪aⅴ在线观看国产 |