本节所谓的“配置同步”主要体现在两个方面:其一,如何监控配置源并在其变化的时候自动加载其数据,其目的是让应用中通过Configuration对象承载的配置与配置源的数据同步;其二、当Configuration对象承载的配置放生变换的时候如何向应用程序发送通知,最终让应用程序使用最新的配置。

一、配置与配置源的同步

配置模型提供了三个原生ConfigurationProvider(JsonConfigrationProvider、XmlConfigurationProvider和IniConfigurationProvider)使我们可以将三种格式(JSON、XML和INI)的文件作为配置原始数据的来源,所以针对物理文件的配置同步是配置同步机制的一个主要的应用领域。在上面演示的实例中,基于物理文件的同步是通过调用ConfigurationRoot的扩展方法ReloadOnChanged来实现的。

这个扩展方法定义在NuGet包“Microsoft.Extensions.Configuration.FileProviderExtensions”之中,除了在我们演示的实例中使用的那个方法之外,这个ReloadOnChanged方法还具有如下两个额外的重载。对于这三个ReloadOnChanged方法重载来说,最终的实现均落在第三个重载上。至于最本质的物理文件监控的功能则由一个名为FileProvider的对象负责。

   1: public static class FileProviderExtensions

   2: {

   3:     public static IConfigurationRoot ReloadOnChanged(

   4:         this IConfigurationRoot config, string filename);

   5:  

   6:     public static IConfigurationRoot ReloadOnChanged(

   7:         this IConfigurationRoot config, string basePath, string filename);

   8:  

   9:     public static IConfigurationRoot ReloadOnChanged(this IConfigurationRoot config, 

  10:         IFileProvider fileProvider, string filename);

  11: }

这里所谓的FileProvider是对所有实现了IFileProvider接口的类型及其对象的统称。IFileProvier接口定义在命名空间“Microsoft.AspNet.FileProviders”下,它通过定义其中的方法提供抽象化的目录与文件信息,针对文件监控相关的方法也定义在这个接口下。如下面的代码片段所示,IFileProvier具有三个方法,其中GetDirectoryContents和GetFileInfo用于提供目录和文件的相关信息,我们只需要关注旨在监控文件变化的Watch方法。

   1: public interface IFileProvider

   2: {

   3:     IDirectoryContents GetDirectoryContents(string subpath);

   4:     IFileInfo GetFileInfo(string subpath);

   5:     IChangeToken Watch(string filter);

   6: }

一个FileProvider总是针对一个具体的目录,Watch方法的参数filter旨在帮助筛选出需要监控的文件。这个参数是一个可以携带通配符(“*”)的字符串,比如 “ *.*”则表示所有文件,而“ *.json”则表示所有扩展名为“ .json”的文件。如果我们需要监控当前目录下某个确定的文件,直接将文件名作为参数即可。Watch方法的返回类型为具有如下定义的IChangeToken接口,我们可以将它理解为一个用于传递数据变换通知的令牌。

   1: public interface IChangeToken

   2: {

   3:     bool HasChanged { get; }

   4:     bool ActiveChangeCallbacks { get; }

   5:     

   6:     IDisposable RegisterChangeCallback(Action<object> callback, object state);    

   7: }

IChangeToken的只读属性HasChanged表示目标数据是否发生改变。我们可以通过调用它的RegisterChangeCallback方法注册一个在数据发生变化时需要执行的回调操作。该方法返回的对象对应的类型必须实现IDisposable接口,回调注册的接触可以通过Dispose方法来完成。至于IChangeToken接口的另个只读属性ActiveChangeCallbacks表示当数据发生变化时是否需要主动执行注册的回调操作。实际上IConfiguration的GetReloadToke方法的返回类型就是这么一个接口,至于该方法具体返回一个怎样的对象,我们会在下一节予以介绍。

当我们指定一个具体的FileProvider对象调用ConfigurationRoot的扩展方法ReloadOnChanged时,后者会调用这个FileProvider的RegisterChangeCallback方法以注册一个在指定文件发生变化时的回调。至于这个注册的回调,它会调用ConfigurationRoot的Reload方法实现对配置数据的重新加载。由于注册了这样一个回调,该方法只需要调用FileProvider的Watch方法监控指定文件的变化即可,如下所示的代码片段基本上体现了ReloadOnChanged方法的逻辑。

   1: public static IConfigurationRoot ReloadOnChanged(

   2:     this IConfigurationRoot config, IFileProvider fileProvider, string filename)

   3: {

   4:     Action<object> callback = null;

   5:     callback = _ =>

   6:     {

   7:         config.Reload();

   8:         fileProvider.Watch(filename).RegisterChangeCallback(callback, null);

   9:     };

  10:     fileProvider.Watch(filename).RegisterChangeCallback(callback, null);

  11:     return config;

  12: }

如果我们通过指定目录和文件名调用另一个ReloadOnChanged方法重载,后者会根据指定的目录创建一个PhysicalFileProvider对象并作为参数调用上面这个重载。顾名思义,PhysicalFileProvider是一个针对具体物理文件的FileProvider,它实际上是借助一个FileSystemWatcher对象来监控指定的文件。这个ReloadOnChanged方法的实现逻辑体现在如下所示的代码片段中。当我们仅仅指定监控文件名调用第一个ReloadOnChanged方法重载时,该方法会将当前应用所在的目录作为参数调用上面一个重载。

   1: public static class FileProviderExtensions

   2: {

   3:    public static IConfigurationRoot ReloadOnChanged(

   4:        this IConfigurationRoot config, string basePath, string filename) 

   5:        => config.ReloadOnChanged(new PhysicalFileProvider(basePath), filename);

   6:     //其他成员

   7: }

二、应用重新加载的配置

ConfigurationRoot通过扩展方法ReloadOnChanged方法与一个具体的物理文件绑定在一起,针对该文件的任何修改操作都会促使Reload方法的调用,进而保证自身承载的数据总是与配置源保持同步。现在我们来讨论配置同步的另一个话题,即如何在不重启应用程序的情况下使用新的配置。要了解这个问题的解决方案,我们得先来聊聊定义在IConfiguration接口中这个一直刻意回避的方法GetReloadToken。

   1: public interface IConfiguration

   2: {

   3:     //其他成员

   4:     IChangeToken GetReloadToken();

   5: }

如上面的代码片段所示,这个GetReloadToken方法的返回类型为上面讨论过的IChangeToken接口,我们说可以将后者视为一个传递数据变化信息的令牌。对于一个Configuration对象来说,它所谓的数据变换体现作为配置根节点的ConfigurationRoot对象的重新加载,所以这个方法返回的ChangeToken对象体现了最近一次加载引起的配置变化。

   1: public class ConfigurationReloadToken : IChangeToken

   2: {

   3:     public void OnReload();

   4:     public IDisposable RegisterChangeCallback(Action<object> callback, 

   5:         object state);

   6:   

   7:     public bool ActiveChangeCallbacks { get; }

   8:     public bool HasChanged { get; }

   9: }

对于实现了IConfiguration接口的两个默认类型(ConfigurationRoot和ConfigurationSection)来说,它们的GetReloadToken方法返回的是一个ConfigurationReloadToken对象。如上面的代码片段所示,除了实现定义在IConfiguration接口中的所有成员之外,ConfigurationReloadToken还具有另一个名为OnReload的方法。当配置数据发生变化,也就是调用通过ConfigurationRoot的Reload方法重新加载配置的时候,这个方法会被调用用以发送“配置已经发生变化”的信号。

实现在ConfigurationReloadToken之中用于传递配置变化的逻辑其实很简单,具体的逻辑是借助于一个CancellationTokenSource对象来完成。如果读者朋友们了解针对Task的异步编程,相信对这个类型不会感到陌生。总的来说,我们可以利用CancellationTokenSource向某个异步执行的Task发送“取消任务”的信号。

   1: public class ConfigurationReloadToken : IChangeToken

   2: {

   3:     private CancellationTokenSource tokenSource = new CancellationTokenSource();

   4:  

   5:     public void OnReload() => tokenSource.Cancel();

   6:     public IDisposable RegisterChangeCallback(Action<object> callback, object state) 

   7:         => tokenSource.Token.Register(callback, state);

   8:  

   9:     public bool ActiveChangeCallbacks { get; } = true;

  10:     public bool HasChanged

  11:     {

  12:         get { return tokenSource.IsCancellationRequested; }

  13:     }

  14: }

如上面的代码片段所示,ConfigurationReloadToken本质上就是一个CancellationTokenSource对象的封装。当OnReload方法被调用的时候,它直接调用CancellationTokenSource的Cancel方法发送取消任务的请求,而HasChanged属性则通过CancellationTokenSource的IsCancellationRequested属性通过判断任务取消请求是否发出来判断配置数据是否发生变化。通过RegisterChangeCallback注册的回调最终注册到由CancellationTokenSource创建的CancellationToken对象上,所以一旦OnReload方法被调用,注册的回调会自动执行。ConfigurationReloadToken的ActiveChangeCallbacks属性总是返回True。

ConfigurationRoot和ConfigurationSection这两个类型分别采用如下的形式实现了GetReloadToken方法。我们从给出的代码片段不难看出所有的ConfigurationSection对象和作为它们根的ConfigurationRoot对象来说,它们的GetReloadToken方法在同一时刻返回的是同一个ConfigurationReloadToken对象。当ConfigurationRoot的Reload方法被调用的时候,当前ConfigurationReloadToken对象的OnReload方法会被调用,在此之后一个新的ConfigurationReloadToken对象会被创建出来并代替原来的对象。

   1: public class ConfigurationRoot : IConfigurationRoot

   2: {

   3:     private ConfigurationReloadToken reloadToken = new ConfigurationReloadToken();

   4:  

   5:     public IChangeToken GetReloadToken()

   6:     {

   7:         return reloadToken;

   8:     }

   9:  

  10:     public void Reload()

  11:     {

  12:         //省略重新加载配置代码

  13:         Interlocked.Exchange<ConfigurationReloadToken>(ref this._reloadToken, 

  14:             new ConfigurationReloadToken()).OnReload();

  15:     }

  16:     //其他成员

  17: }

  18:  

  19: public class ConfigurationSection : IConfigurationSection, IConfiguration

  20: {

  21:     private readonly ConfigurationRoot root;

  22:     public IChangeToken GetReloadToken()

  23:     {

  24:         return root.GetReloadToken();

  25:     }

  26:     //其他成员

  27: }

正是因为GetReloadToken方法并不能保证每次返回的都是同一个ConfigurationReloadToken对象,所以当我们注册配置加载回调时,需要在回调中完成针对新的ConfigurationReloadToken对象的回调注册,实际上我们上面演示的实例就是这么做的。除此之外,调用RegisterChangeCallback方法会返回一个类型实现了IDisposable 接口的对象,不要忘记调用它的Dispose方法以免产生一些内存泄漏的问题。

   1: public class Program

   2: {

   3:     private static IDisposable callbackRegistration;

   4:     private static void OnSettingChanged(object state)

   5:     {

   6:         callbackRegistration?.Dispose();

   7:         IConfiguration configuration = (IConfiguration)state;

   8:         Console.WriteLine(configuration.Get<ThreadPoolSettings>());

   9:         callbackRegistration = configuration.GetReloadToken()

  10:             .RegisterChangeCallback(OnSettingChanged, state);

  11:     }

  12: }

ASP.NET Core的配置(1):读取配置信息
ASP.NET Core的配置(2):配置模型详解
ASP.NET Core的配置(3): 将配置绑定为对象[上篇]
ASP.NET Core的配置(3): 将配置绑定为对象[下篇]
ASP.NET Core的配置(4):多样性的配置源[上篇]
ASP.NET Core的配置(4):多样性的配置源[中篇]
ASP.NET Core的配置(4):多样性的配置源[下篇]
ASP.NET Core的配置(5):配置的同步[上篇]
ASP.NET Core的配置(5):配置的同步[下篇]

ASP.NET Core的配置(5):配置的同步[设计篇]的更多相关文章

  1. 跟我学: 使用 fireasy 搭建 asp.net core 项目系列之三 —— 配置

    ==== 目录 ==== 跟我学: 使用 fireasy 搭建 asp.net core 项目系列之一 —— 开篇 跟我学: 使用 fireasy 搭建 asp.net core 项目系列之二 —— ...

  2. [ASP.NET Core 3框架揭秘] 配置[1]:读取配置数据[上篇]

    提到"配置"二字,我想绝大部分.NET开发人员脑海中会立即浮现出两个特殊文件的身影,那就是我们再熟悉不过的app.config和web.config,多年以来我们已经习惯了将结构化 ...

  3. [ASP.NET Core 3框架揭秘] 配置[2]:读取配置数据[下篇]

    [接上篇]提到“配置”二字,我想绝大部分.NET开发人员脑海中会立即浮现出两个特殊文件的身影,那就是我们再熟悉不过的app.config和web.config,多年以来我们已经习惯了将结构化的配置定义 ...

  4. [ASP.NET Core 3框架揭秘] 配置[5]:配置数据与数据源的实时同步

    在<配置模型总体设计>介绍配置模型核心对象的时候,我们刻意回避了与配置同步相关的API,现在我们利用一个独立文章来专门讨论这个话题.配置的同步涉及到两个方面:第一,对原始的配置源实施监控并 ...

  5. [ASP.NET Core 3框架揭秘] 配置[3]:配置模型总体设计

    在<读取配置数据>([上篇],[下篇])上面一节中,我们通过实例的方式演示了几种典型的配置读取方式,接下来我们从设计的维度来重写认识配置模型.配置的编程模型涉及到三个核心对象,分别通过三个 ...

  6. [ASP.NET Core 3框架揭秘] 配置[7]:多样化的配置源[中篇]

    物理文件是我们最常用到的原始配置载体,而最佳的配置文件格式主要有三种,它们分别是JSON.XML和INI,对应的配置源类型分别是JsonConfigurationSource.XmlConfigura ...

  7. [ASP.NET Core 3框架揭秘] 配置[6]:多样化的配置源[上篇]

    .NET Core采用的这个全新的配置模型的一个主要的特点就是对多种不同配置源的支持.我们可以将内存变量.命令行参数.环境变量和物理文件作为原始配置数据的来源.如果采用物理文件作为配置源,我们可以选择 ...

  8. [ASP.NET Core 3框架揭秘] 配置[4]:将配置绑定为对象

    虽然应用程序可以直接利用通过IConfigurationBuilder对象创建的IConfiguration对象来提取配置数据,但是我们更倾向于将其转换成一个POCO对象,以面向对象的方式来使用配置, ...

  9. 【ASP.NET Core】自定义的配置源

    本文的主题是简单说说如何实现 IConfigurationSource.IConfigurationProvider 接口来自定义一个配置信息的来源,后面老周给的示例是实现用 CSV 文件进行应用配置 ...

随机推荐

  1. ASP.NET Aries 入门开发教程6:列表数据表格的格式化处理及行内编辑

    前言: 为了赶进度,周末也写文了! 前几篇讲完查询框和工具栏,这节讲表格数据相关的操作. 先看一下列表: 接下来我们有很多事情可以做. 1:格式化 - 键值的翻译 对于“启用”列,已经配置了格式化 # ...

  2. SQLServer执行命令出现“目录无效的提示”

    异常处理汇总-数据库系列  http://www.cnblogs.com/dunitian/p/4522990.html 一般都是清理垃圾清理过头了,把不该删的目录删了 网上说法: 问题描述: 1.s ...

  3. .NET中AOP方便之神SheepAspect

    SheepAspect 简介以及代码示列: SheepAspect是一个AOP框架为.NET平台,深受AspectJ.它静织目标组件作为一个编译后的任务(编译时把AOP代码植入). 多有特性时,可根据 ...

  4. iOS网络4——Reachability检测网络状态

    一.整体介绍 前面已经介绍了网络访问的NSURLSession.NSURLConnection,还有网页加载有关的webview,基本满足通常的网络相关的开发. 其实在网络开发中还有比较常用的就是网络 ...

  5. git提交项目到已存在的远程分支

    今天想提交项目到github的远程分支上,那个远程分支是之前就创建好的,而我的本地关联分支还没创建.   之前从未用github提交到远程分支过,弄了半个钟,看了几篇博文,终于折腾出来.现在把步骤整理 ...

  6. Atitit 管理原理与实践attilax总结

    Atitit 管理原理与实践attilax总结 1. 管理学分类1 2. 我要学的管理学科2 3. 管理学原理2 4. 管理心理学2 5. 现代管理理论与方法2 6. <领导科学与艺术4 7. ...

  7. Impress.js上手 - 抛开PPT、制作Web 3D幻灯片放映

    前言: 如果你已经厌倦了使用PPT设置路径.设置时间.设置动画方式来制作动画特效.那么Impress.js将是你一个非常好的选择. 用它制作的PPT将更加直观.效果也是嗷嗷美观的. 当然,如果用它来装 ...

  8. test

    http://img.ivsky.com/img/bizhi/pic/201009/07/fangaoyouhua-015.jpghttp://desk.fd.zol-img.com.cn/t_s16 ...

  9. CYQ.Data V5 分布式缓存Redis应用开发及实现算法原理介绍

    前言: 自从CYQ.Data框架出了数据库读写分离.分布式缓存MemCache.自动缓存等大功能之后,就进入了频繁的细节打磨优化阶段. 从以下的更新列表就可以看出来了,3个月更新了100条次功能: 3 ...

  10. ASP.NET Identity入门系列教程(一) 初识Identity

    摘要 通过本文你将了解ASP.NET身份验证机制,表单认证的基本流程,ASP.NET Membership的一些弊端以及ASP.NET Identity的主要优势. 目录 身份验证(Authentic ...