欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

.Net Core自定義配置源從配置中心讀取配置的方法

 更新時間:2019年03月06日 09:06:53   作者:Agile.Zhou  
這篇文章主要給大家介紹了關(guān)于.Net Core自定義配置源從配置中心讀取配置的相關(guān)資料,文中通過示例代碼以及圖文介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧

前言

配置,幾乎所有的應(yīng)用程序都離不開它。.Net Framework時代我們使用App.config、Web.config,到了.Net Core的時代我們使用appsettings.json,這些我們再熟悉不過了。然而到了容器化、微服務(wù)的時代,這些本地文件配置有的時候就不太合適了。當(dāng)你把本地部署的服務(wù)搬到docker上后,你會發(fā)現(xiàn)要修改一個配置文件變的非常麻煩。你不得不通過宿主機進(jìn)入容器內(nèi)部來修改文件,也許容器內(nèi)還不帶vi等編輯工具,你連看都不能看,改都不能。更別說當(dāng)你啟動多個容器實例來做分布式應(yīng)用的時候,一個個去修改容器的配置,這簡直要命了。

因為這些原因,所以“配置中心”就誕生了。配置中心是微服務(wù)的基礎(chǔ)設(shè)施,它對配置進(jìn)行集中的管理并對外暴露接口,當(dāng)應(yīng)用程序需要的時候通過接口讀取。配置通常為Key/Value模式,然后通過http接口暴露。好了,配置中心不多說了,感覺要偏了,這次是介紹怎么自定義一個配置源從配置中心讀取配置。廢話不多說直接上代碼吧。

模擬配置中心

我們新建一個asp.net core webapi站點來模擬配置中心服務(wù),端口配置到5000,并添加相應(yīng)的controller來模擬配置中心對外的接口。

  [Route("api/[controller]")]
 [ApiController]
 public class ConfigsController : ControllerBase
 {
  public List<KeyValuePair<string,string>> Get()
  {
   var configs = new List<KeyValuePair<string, string>>();
   configs.Add(new KeyValuePair<string, string>("SecretKey","1238918290381923"));
   configs.Add(new KeyValuePair<string, string>("ConnectionString", "user=123;password=123;server=."));

   return configs;
  }
 }

添加一個configscontroller,并修改Get方法,返回2個配置鍵值對。

訪問下/api/configs看下返回是否正確

自定義配置源

從現(xiàn)在開始我們真正開始來定義一個自定義的配置源然后當(dāng)程序啟動的時候從配置中心讀取配置文件信息,并提供給后面的代碼使用配置。

新建一個asp.net core mvc站點來模擬客戶端程序。

MyConfigProvider

 public class MyConfigProvider : ConfigurationProvider
 {
  /// <summary>
  /// 嘗試從遠(yuǎn)程配置中心讀取配置信息
  /// </summary>
  public async override void Load()
  {
   var response = "";
   try
   {
    var serverAddress = "http://localhost:5000";
    var client = new HttpClient();
    client.BaseAddress = new Uri(serverAddress);
    response = await client.GetStringAsync("/api/configs");
   }
   catch (Exception ex)
   {
    //write err log
   }

   if (string.IsNullOrEmpty(response))
   {
    throw new Exception("Can not request configs from remote config center .");
   }

   var configs = JsonConvert.DeserializeObject<List<KeyValuePair<string, string>>>(response);

   Data = new ConcurrentDictionary<string, string>();

   configs.ForEach(c =>
   {
    Data.Add(c);
   });
  }
 
 }

新建一個MyConfigProvider的類,這個類從ConfigurationProvider繼承,并重寫其中的Load方法。使用HttpClient從配置中心讀取信息后,進(jìn)行反序列化,并把配置轉(zhuǎn)換為字典。這里注意一下,雖然Data的類型為IDictionary<string,string>,但是這里實例化對象的時候使用了ConcurrentDictionary<string, string>類,因為Dictionary<string,string>是非線程安全的,如果進(jìn)行多線程讀寫會出問題。

MyConfigSource

 public class MyConfigSource : IConfigurationSource
 {
  public IConfigurationProvider Build(IConfigurationBuilder builder)
  {
   return new MyConfigProvider();
  }
 }

新建一個MyConfigSource的類,這個類實現(xiàn)IConfigurationSource接口,IConfigurationSource接口只有一個Build方法,返回值為IConfigurationProvider,我們剛才定義的MyConfigProvider因為繼承自ConfigurationProvider所以已經(jīng)實現(xiàn)了IConfigurationProvider,我們直接new一個MyConfigProvider并返回。

MyConfigBuilderExt

 public static class MyConfigBuilderExt
 {
  public static IConfigurationBuilder AddMyConfig(
   this IConfigurationBuilder builder
   )
  {
   return builder.Add(new MyConfigSource());
  }
 }

給IConfigurationBuilder定義一個AddMyConfig的擴展方法,跟.Net Core自帶的幾個配置源使用風(fēng)格保持一致。當(dāng)調(diào)用AddMyConfig的時候給IConfigurationBuilder實例添加一個MyConfigSource的源。

使用配置源

在Program中添加MyConfigSource

  public class Program
 {
  public static void Main(string[] args)
  {
   CreateWebHostBuilder(args).Build().Run();
  }

  public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
   WebHost.CreateDefaultBuilder(args)
   .ConfigureAppConfiguration((context, configBuiler) =>
   {
    configBuiler.AddMyConfig();
   })
   .UseStartup<Startup>();
 }

在ConfigureAppConfiguration的匿名委托方法中調(diào)用AddMyConfig擴展方法,這樣程序啟動的時候會自動使用MyConfigSource源并從配置中心讀取配置到本地應(yīng)用程序。

修改HomeController

 public class HomeController : Controller
 {
  IConfiguration _configuration;
  public HomeController(IConfiguration configuration)
  {
   _configuration = configuration;
  }

  public IActionResult Index()
  {
   var secretKey = _configuration["SecretKey"];
   var connectionString = _configuration["ConnectionString"];

   ViewBag.SecretKey = secretKey;
   ViewBag.ConnectionString = connectionString;

   return View();
  }
  
 }

修改homecontroller,把IConfiguration通過構(gòu)造函數(shù)注入進(jìn)去,在Index Action方法中讀取配置,并賦值給ViewBag

修改Index視圖

 @{
 ViewData["Title"] = "Test my config";
}

<h3>
 SecretKey: @ViewBag.SecretKey
</h3>
<h3>
 ConnectionString: @ViewBag.ConnectionString
</h3>

修改Index視圖的代碼,把配置信息從ViewBag中讀取出來并在網(wǎng)頁上展示。

運行一下

先運行配置中心站點再運行一下網(wǎng)站,首頁出現(xiàn)了我們在配置中心定義的SecretKey跟ConnectionString信息,表示我們的程序成功的從配置中心讀取了配置信息。我們的自定義配置源已經(jīng)能夠成功運行了。

改進(jìn)

以上配置源雖然能夠成功運行,但是仔細(xì)看的話顯然它有2個比較大的問題。

  • 配置中心的服務(wù)地址是寫死在類里的。我們的配置中心很有可能會修改ip或者域名,寫死在代碼里顯然不是高明之舉,所以我們還是需要保留本地配置文件,把配置中心的服務(wù)地址寫到本地配置文件中。
  • 配置中心作為微服務(wù)的基礎(chǔ)設(shè)施一旦故障會引發(fā)非常嚴(yán)重的后果,新啟動或者重啟的客戶端會無法正常啟動。如果我們在配置中心正常的時候冗余一份配置在本地,當(dāng)配置中心故障的時候從本地讀取配置,至少可以保證一部分客戶端程序能夠正常運行。
{
 "Logging": {
 "LogLevel": {
  "Default": "Warning"
 }
 },
 "AllowedHosts": "*",
 "myconfigServer": "http://localhost:5000"
}

修改本地appsettings.json文件,添加myconfigServer的配置信息。

 public class MyConfigProvider : ConfigurationProvider
 {
  private string _serverAddress;
  public MyConfigProvider()
  {
   var jsonConfig = new JsonConfigurationSource();
   jsonConfig.FileProvider = new PhysicalFileProvider(Directory.GetCurrentDirectory());
   jsonConfig.Path = "appsettings.json";
   var jsonProvider = new JsonConfigurationProvider(jsonConfig);
   jsonProvider.Load();

   jsonProvider.TryGet("myconfigServer", out string serverAddress);

   if (string.IsNullOrEmpty(serverAddress))
   {
    throw new Exception("Can not find myconfigServer's address from appsettings.json");
   }

   _serverAddress = serverAddress;
  }

  /// <summary>
  /// 嘗試從遠(yuǎn)程配置中心讀取配置信息,當(dāng)成功從配置中心讀取信息的時候把配置寫到本地的myconfig.json文件中,當(dāng)配置中心無法訪問的時候嘗試從本地文件恢復(fù)配置。
  /// </summary>
  public async override void Load()
  {
   var response = "";
   try
   {
    var client = new HttpClient();
    client.BaseAddress = new Uri(_serverAddress);
    response = await client.GetStringAsync("/api/configs");

    WriteToLocal(response);
   }
   catch (Exception ex)
   {
    //write err log
    response = ReadFromLocal();
   }

   if (string.IsNullOrEmpty(response))
   {
    throw new Exception("Can not request configs from remote config center .");
   }

   var configs = JsonConvert.DeserializeObject<List<KeyValuePair<string, string>>>(response);

   Data = new ConcurrentDictionary<string, string>();

   configs.ForEach(c =>
   {
    Data.Add(c);
   });
  }

  private void WriteToLocal(string resp)
  {
   var file = Directory.GetCurrentDirectory() + "/myconfig.json";
   File.WriteAllText(file,resp);
  }

  private string ReadFromLocal()
  {
   var file = Directory.GetCurrentDirectory() + "/myconfig.json";
   return File.ReadAllText(file);
  }
 }

修改MyConfigProvider,修改構(gòu)造函數(shù),通過JsonConfigurationProvider從本地讀取appsettings.json中的myconfigServer配置信息。新增WriteToLocal方法把配置中心返回的json數(shù)據(jù)寫到本地文件中。新增ReadFromLocal方法,從本地文件讀取json信息。

再次運行

先運行配置中心站點,再運行客戶端網(wǎng)站,可以看到配置信息展示到首頁界面上。關(guān)閉配置中心客跟客戶端網(wǎng)站,并且重啟客戶端網(wǎng)站依然能夠展示配置信息,說明自定義配置源當(dāng)配置中心故障的時候成功從本地文件恢復(fù)了配置。圖跟上面的圖是一致的,就不貼了。

總結(jié)

通過以上我們定義了一個比較簡單的自定義配置源,它能夠通過http從配置中心讀取配置,并且提供了同傳統(tǒng)json配置文件一致的使用風(fēng)格,最大程度的復(fù)用舊代碼,減少因為引入配置中心而大規(guī)模改動代碼。我們從上面的代碼可以更清楚的知道.Net Core的配置源是如何工作的。ConfigurationSource只是ConfigurationProvider的建造器。真正完成配置加載、查找工作的是ConfigurationProvider。

以上代碼還是演示級別的代碼,還有很多改進(jìn)的空間,比如http訪問失敗的重試,我們可以使用polly重構(gòu);比如支持定時從配置中心刷新配置等,有興趣可以自己去實踐一下。

好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關(guān)文章

  • UrlRewrite概念原理及使用方法解析

    UrlRewrite概念原理及使用方法解析

    這篇文章主要介紹了UrlRewrite概念原理及使用方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-10-10
  • .Net Core 集成 Kafka的步驟

    .Net Core 集成 Kafka的步驟

    這篇文章主要介紹了.Net Core 集成 Kafka的實現(xiàn)步驟,幫助大家更好的理解和學(xué)習(xí)使用.net技術(shù),感興趣的朋友可以了解下
    2021-04-04
  • .Net6.0+Vue3實現(xiàn)數(shù)據(jù)簡易導(dǎo)入功能全過程

    .Net6.0+Vue3實現(xiàn)數(shù)據(jù)簡易導(dǎo)入功能全過程

    最近在用VUE做一個數(shù)據(jù)導(dǎo)入的功能,下面這篇文章主要給大家介紹了關(guān)于使用.Net6.0+Vue3實現(xiàn)數(shù)據(jù)簡易導(dǎo)入功能的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-09-09
  • webapi跨域使用session的方法示例

    webapi跨域使用session的方法示例

    這篇文章主要介紹了webapi跨域使用session的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-07-07
  • ASP.NET郵件發(fā)送system.Net.Mail案例

    ASP.NET郵件發(fā)送system.Net.Mail案例

    這篇文章主要為大家詳細(xì)介紹了ASP.NET郵件發(fā)送system.Net.Mail案例,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-05-05
  • Entity Framework Core表名映射

    Entity Framework Core表名映射

    這篇文章介紹了Entity Framework Core實現(xiàn)表名映射的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-03-03
  • ASP.NET下使用WScript.Shell執(zhí)行命令

    ASP.NET下使用WScript.Shell執(zhí)行命令

    ASP.NET下有自己的執(zhí)行CMD命令的方式,這里用WScript.Shell似有畫蛇添足之嫌,但是我們也不能排除真的有機器禁用了.NET的相關(guān)類,未雨綢繆嘛。當(dāng)然也不僅僅局限于WScript.Shell,只要是ASP中能用的組件,統(tǒng)統(tǒng)都可以用于ASP.NET中,而且還更方便!
    2008-05-05
  • GridView使用學(xué)習(xí)總結(jié)

    GridView使用學(xué)習(xí)總結(jié)

    這篇文章主要是關(guān)于GridView使用的學(xué)習(xí)總結(jié),需要的朋友可以參考下
    2015-09-09
  • ASP.Net獲取客戶端網(wǎng)卡MAC的小例子

    ASP.Net獲取客戶端網(wǎng)卡MAC的小例子

    這篇文章介紹了ASP.Net獲取客戶端網(wǎng)卡MAC的小例子,有需要的朋友可以參考一下
    2013-07-07
  • 詳解Asp.net web.config customErrors 如何設(shè)置

    詳解Asp.net web.config customErrors 如何設(shè)置

    這篇文章主要介紹了詳解Asp.net web.config customErrors 如何設(shè)置,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-02-02

最新評論