asp.net core重新加載應(yīng)用配置

asp.net core重新加載應(yīng)用配置

Intro

我把配置放在了數(shù)據(jù)庫或者是Redis里,配置需要修改的時(shí)候我要直接修改數(shù)據(jù)庫,然后調(diào)用一個(gè)接口去重新加載應(yīng)用配置,于是就嘗試寫一個(gè)運(yùn)行時(shí)重新加載配置的接口。

Configuration 重新加載實(shí)現(xiàn)

重新加載配置的接口其實(shí)很簡單,經(jīng)過看 Configuration 的源碼可以知道,如果想要重新加載應(yīng)用配置,需要一個(gè) IConfigurationRoot 對象,而 IConfigurationRoot 其實(shí)可以直接拿注入服務(wù)中的 IConfiguration 對象,服務(wù)中的 IConfiguration 對象也是實(shí)現(xiàn)了 IConfigurationRoot 接口的實(shí)例。后面我們一起看源碼就更清晰了。

來看實(shí)現(xiàn)重新加載配置的代碼

using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Configuration;

namespace TestWebApplication.Controllers
{
    [Route("api/[controller]")]
    public class ConfigurationsController : Controller
    {
        private readonly IConfigurationRoot _configuration;

        public ConfigurationsController(IConfiguration configuration)
        {
            _configuration = configuration as IConfigurationRoot;
        }

        [HttpGet]
        public IActionResult Get()
        {
            return Ok(new
            {
                RootUser = _configuration.GetAppSetting("RootUser") // 這里 GetAppSetting 是一個(gè)自定義擴(kuò)展方法,獲取AppSettings 節(jié)點(diǎn)下的配置信息
            });
        }

        [HttpPut]
        public IActionResult Put()
        {
            _configuration.Reload();
            return Ok();
        }
    }
}

精簡版

        /// <summary>
        /// 重新加載系統(tǒng)配置
        /// </summary>
        /// <returns></returns>
        public IActionResult ReloadConfiguration()
        {
            var configurationRoot = HttpContext.RequestServices.GetService<IConfiguration>() as IConfigurationRoot;
            if (null == configurationRoot)
            {
                return BadRequest();
            }
            configurationRoot.Reload();
            return Ok();
        }

是不是很簡單,下面我們來嘗試一下,你可以參考這個(gè)示例項(xiàng)目

因?yàn)槟J(rèn)的項(xiàng)目配置會監(jiān)聽 appsettings.json 文件是否修改,如果已修改就會重新reload,這里我新加一個(gè)文件,這里設(shè)置 reloadOnChangefalse,示例代碼如下:

public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
            WebHost.CreateDefaultBuilder(args)
                .ConfigureAppConfiguration(configBuilder =>
                {
                    configBuilder.AddJsonFile("abc.json", optional: true, reloadOnChange: false);
                })
                .UseStartup<Startup>();

abc.json 的文件內(nèi)容如下:

{
  "AppSettings": {
    "TestNumber": 12,
    "RootUser": "WeihanLi"
  }
}

dotnet run 啟動網(wǎng)站,然后在瀏覽器中訪問 http://localhost:5000/api/configurations

然后我們修改 abc.json 文件

{
  "AppSettings": {
    "TestNumber": 12,
    "RootUser": "WeihanLi 123"
  }
}

修改保存之后刷新剛才的頁面,可以看到還是剛才的內(nèi)容,證明并沒有重新加載配置,接下來嘗試我們的重新加載配置方法

使用 postman 或 fiddler 或其他你喜歡的工具發(fā)一個(gè) PUT 請求到 http://localhost:5000/api/configurations,這里我使用 postman 調(diào)用 PUT 接口重新加載配置

reload configuration

返回 200 即接口調(diào)用成功,重新刷新剛才的頁面就可以看到頁面上的數(shù)據(jù)已經(jīng)發(fā)生變化,這也就證明了我們重新加載配置的接口生效了。

源碼解析

來看 ConfigurationBuilder 在 Build 的時(shí)候做了什么,ConfigurationBuilder 源碼

ConfigurationBuilder.Build()

可以看到這里最后返回的是一個(gè) IConfigurationRoot 對象,再來看 IConfigurationRoot 源碼

IConfigurationRoot

可以看到 IConfigurationRoot 定義了一個(gè) Reload 的方法,這個(gè)方法會從下面的 Providers 中重新加載配置,看到這里我們就知道可以通過 IConfigurationReload 方法來重新加載應(yīng)用程序的配置了,然后我們來看 WebHost.CreateDefaultBuilder(args).Build() 做了什么
https://github.com/aspnet/AspNetCore/blob/master/src/DefaultBuilder/src/WebHost.cs#L149

WebHost

這里我們可以看到為什么 appsettings.json 文件會自動 reload 配置,可以看到最后返回了一個(gè) WebHostBuilder 對象

看 Asp.Net core WebHostBuilder 對象的 Build 方法 https://github.com/aspnet/AspNetCore/blob/master/src/Hosting/Hosting/src/WebHostBuilder.cs#L135

BuildCommonServices 可以看到這樣一段代碼 https://github.com/aspnet/AspNetCore/blob/master/src/Hosting/Hosting/src/WebHostBuilder.cs#L277

image.png

上面我們已經(jīng)知道 ConfigurationBuilder Build 之后返回的是一個(gè) IConfigurationRoot 對象,而這里注入是一個(gè) IConfiguration 對象(IConfigurationRoot 實(shí)現(xiàn) IConfiguration 接口),所以我們就可以從依賴注入中獲取 IConfiguration 對象直接當(dāng)作 IConfigurationRoot 來使用,這也就是為什么我們會直接獲取一個(gè) IConfiguration 對象直接 as IConfigurationRoot

Memo

到此就暫時(shí)結(jié)束了,希望你能有所收獲~

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容