Options是.net core提出的一种辅助配置机制,即选项。

  目前,我们可以使用的Options有五种(源码):

  IOptionsFactory<>:Options的创建工厂(Singleton),所有的Options均使用它仅限创建,可以使用名称创建指定名称的Options

  IOptions<>:常用的一种Options模式,不可指定名称,且会缓存Options,全局将使用这个缓存的Options,没有重新加载机制,不可修改。(ps:可使用反射进行修改)

  IOptionsSnapshot<>:和IOptionsFactory<>差不多,内部使用IOptionsFactory<>来创建,但是IOptionsSnapshot<>内部有缓存,且它的作用于是Scoped,做web开发的时候,如果不想自己实现重新加载,不妨使用IOptionsSnapshot<>。

  IOptionsMonitor<>和IOptionsMonitorCache<>:IOptionsMonitorCache<>是IOptionsMonitor<>的缓存,读取IOptionsMonitor<>时,会先从IOptionsMonitorCache<>缓存中读取,没有则使用IOptionsFactory<>创建并加入到IOptionsMonitorCache<>缓存中去,因此当配置发生改变时,我们需要重新加载才能得到新的Options,所谓重新加载,就是Options中的数据重新执行Configure以及PostConfigure中的方法。

  在继续下面的内容之前,我们应该明确,只有IOptionsMonitor<>会涉及到重新加载,那么何时需要重新加载?也就是何时触发这个重新加载?不同的系统做法会不一样,有的使用一条消息总线,有的使用消息的发布订阅,有的使用接口的手动触发,本文介绍采用定时器来模拟说明:

  一、使用IOptionsMonitorCache<>

  上面介绍到,IOptionsMonitorCache<>是IOptionsMonitor<>缓存,自然可以通过IOptionsMonitorCache<>来清除缓存来实现Options的重新创建了,如:

  假如数据源是一个类的数据,然后使用定时器定时刷新,刷新后需要清除IOptionsMonitorCache<>缓存:  

    //表示Options的数据来源
public class OptionsSource
{
public static DateTime Time { get; private set; } public OptionsSource(IServiceProvider serviceProvider)
{
Timer timer = new Timer();
timer.Interval = 10000;//10秒钟刷新一次Time
timer.Elapsed += new ElapsedEventHandler((o, e) =>
{
Time = DateTime.Now; //清除缓存
var options = serviceProvider.GetService(typeof(IOptionsMonitorCache<DemoOptions>)) as IOptionsMonitorCache<DemoOptions>;
options.Clear();//或者使用TryRemove删除某个名称的Options
});
timer.Start();//启动
}
}

  接着在Startup中注册Options:  

    public void ConfigureServices(IServiceCollection services)
{
//注册Options
services.Configure<DemoOptions>(options =>
{
options.Time = OptionsSource.Time;
});
services.AddSingleton<OptionsSource>();//数据源 //其他服务注入代码
} public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
app.ApplicationServices.GetRequiredService<OptionsSource>();//获取一次,启动定时器 //其他代码
}

  接下来可以使用IOptionsMonitor<>获取Options,会发现每隔10秒,数据会刷新一次,比如有一个接口读取Options,每10秒调用试试:  

    [HttpGet]
public object GetOptions()
{
var options = HttpContext.RequestServices.GetService(typeof(IOptionsMonitor<DemoOptions>)) as IOptionsMonitor<DemoOptions>;
return options.CurrentValue;
}

  二、使用IOptionsChangeTokenSource<>

  IOptionsChangeTokenSource<>是IOptionsMonitor<>用于触发重新加载Options的接口,它有一个Name属性,表示重新加载某个指定名称的Options,还有一个GetChangeToken方法,用于获取一个IChangeToken,由这个IChangeToken来触发何时应该重新加载Options,比如:

  我们先实现IOptionsChangeTokenSource<>接口:

    public class MyOptionsChangeTokenSource<TOptions> : IOptionsChangeTokenSource<TOptions> where TOptions : class
{
ConfigurationReloadToken _reloadToken = new ConfigurationReloadToken(); public MyOptionsChangeTokenSource(string name)
{
Name = name ?? Options.DefaultName;
} public string Name { get; } public IChangeToken GetChangeToken()
{
return _reloadToken;
}
public void OnReload()
{
var previousToken = System.Threading.Interlocked.Exchange(ref _reloadToken, new ConfigurationReloadToken());
previousToken.OnReload();
}
}

  其中OnReload()方法即触发重新加载,何时触发呢?假如配置的数据源来自某个类的数据:  

    //表示Options的数据来源
public class OptionsSource
{
public static DateTime Time { get; private set; } public OptionsSource(IOptionsChangeTokenSource<DemoOptions> optionsChangeTokenSource)
{
Timer timer = new Timer();
timer.Interval = 10000;//10秒钟刷新一次Time
timer.Elapsed += new ElapsedEventHandler((o, e) =>
{
Time = DateTime.Now; if (optionsChangeTokenSource is MyOptionsChangeTokenSource<DemoOptions> source)
{
source.OnReload();//触发MyOptionsChangeTokenSource<>从而重新加载Options
}
});
timer.Start();//启动
}
}

  接着在Startup中注册Options:  

    public void ConfigureServices(IServiceCollection services)
{
//注册Options
services.Configure<DemoOptions>(options =>
{
options.Time = OptionsSource.Time;
});
services.AddSingleton<OptionsSource>();//数据源
services.AddSingleton<IOptionsChangeTokenSource<DemoOptions>>(new MyOptionsChangeTokenSource<DemoOptions>("")); //其他代码
} // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
app.ApplicationServices.GetRequiredService<OptionsSource>();//获取一次,启动定时器 //其他代码
}

  接下来可以使用IOptionsMonitor<>获取Options也会有重新加载的效果了。

  上面的例子中,数据源来自某个类,但是开发过程中,我们的Options中的数据往往来自IConfiguration配置中,这时就更简单了,框架已经给我们准备好了一个ConfigurationChangeTokenSource<>类,当我们使用IConfiguration来配置Options时会自动注入这个类,我们当然也可以手动注入,意思就是说,当对应的IConfiguration中的数据发生改变时,它会触发清除对应的IOptionsMonitorCache<>缓存从而导致重新加载,比如:

  appsettings.json中有下面的json片段:  

  {
"Demo": {
"Value": 1
}
}

  程序启动时会自动加载appsettings.json文件中的配置到IConfiguration中,所以我们可以直接从IConfiguration配置Options:  

    public void ConfigureServices(IServiceCollection services)
{
//方式一:
//这个会添加ConfigurationChangeTokenSource<>类,程序启动后,修改appsettings.json会触发Options的重新加载
services.Configure<DemoOptions>(Configuration.GetSection("Demo")); //方式二:
//这种方式不会添加ConfigurationChangeTokenSource<>类,但是可以手动添加
services.Configure<DemoOptions>(options =>
{
Configuration.GetSection("Demo").Bind(options);
});
services.AddSingleton<IOptionsChangeTokenSource<DemoOptions>>(new ConfigurationChangeTokenSource<DemoOptions>(Options.DefaultName, Configuration.GetSection("Demo"))); //其他代码
}

  如上所述,添加Options的Configure方法有很多,但是只用 OptionsConfigurationServiceCollectionExtensions 类中的Configure方法才会自动添加ConfigurationChangeTokenSource<>类,即Configure方法参数中有IConfiguration参数。

  结语

  Options的重新加载,如果看看源码就会很清晰,但是更多的,希望在读取Options的时候多注意一下吧:

  1、如果配置永远不会发生改变,或者改变或需要重启才能生效(这种配置往往是在启动阶段用到),那么尽量使用IOptions<>

  2、如果要实现配置修改后会自动加载Options,推荐使用OptionsMonitor<>,但是注意,要自行实现何时需要重新加载,其次可以使用IOptionsSnapshot<>,因为它是一个Scoped作用域,最次是IOptionsFactory<>,每次都是重新创建可能会导致一些性能的损失

.net core中的Options重新加载机制的更多相关文章

  1. Android之Android apk动态加载机制的研究(二):资源加载和activity生命周期管理

    转载请注明出处:http://blog.csdn.net/singwhatiwanna/article/details/23387079 (来自singwhatiwanna的csdn博客) 前言 为了 ...

  2. EF如何操作内存中的数据以及加载相关联表的数据:延迟加载、贪婪加载、显示加载

    之前的EF Code First系列讲了那么多如何配置实体和数据库表的关系,显然配置只是辅助,使用EF操作数据库才是每天开发中都需要用的,这个系列讲讲如何使用EF操作数据库.老版本的EF主要是通过Ob ...

  3. .Net Core 程序集管理说明(加载)

    .NET CORE 的程序集加载管理和以前的 .NET 发生了很大的变化, 在 .NET CORE 里, 程序集的加载, 依赖了 xx.deps.json 文件, deps.json 文件里,定义了程 ...

  4. imagesLoaded – 检测网页中的图片是否加载

    imagesLoaded 是一个用于来检测网页中的图片是否载入完成的 JavaScript 工具库.支持回调的获取图片加载的进度,还可以绑定自定义事件.可以结合 jQuery.RequireJS 使用 ...

  5. 在Unity3D的网络游戏中实现资源动态加载

    用Unity3D制作基于web的网络游戏,不可避免的会用到一个技术-资源动态加载.比如想加载一个大场景的资源,不应该在游戏的开始让用户长时间等待全部资源的加载完毕.应该优先加载用户附近的场景资源,在游 ...

  6. JVM中java类的加载时机(转载:http://blog.csdn.net/chenleixing/article/details/47099725)

    Java虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的加载机制.类从被加载到虚拟机内存中开始,到卸载出内 ...

  7. 动态加载JS过程中如何判断JS加载完成

    在正常的加载过程中,js文件的加载是同步的,也就是说在js加载的过程中,浏览器会阻塞接下来的内容的解析.这时候,动态加载便显得尤为重要了,由于它是异步加载,因此,它可以在后台自动下载,并不会妨碍其它内 ...

  8. MapXtreme在asp.net中的使用之加载地图(转)

    MapXtreme在asp.net中的使用之加载地图(转) Posted on 2010-05-04 19:44 Happy Coding 阅读(669) 评论(0) 编辑 收藏 1.地图保存在本地的 ...

  9. Qt中如何 编写插件 加载插件 卸载插件

    Qt中如何 编写插件 加载插件 卸载插件是本文要介绍的内容.Qt提供了一个类QPluginLoader来加载静态库和动态库,在Qt中,Qt把动态库和静态库都看成是一个插件,使用QPluginLoade ...

随机推荐

  1. redis入门到精通系列(一)

    (一)为什么要用Nosql 如果你是计算机本科学生 ,那么一定使用过关系型数据库mysql.在请求量小的情况下,使用mysql不会有任何问题,但是一旦同时有成千上万个请求同时来访问系统时,就会出现卡顿 ...

  2. shell获取目录下(包括子目录)所有文件名、路径、文件大小

    一例shell脚本:取得目录下(包括子目录)所有文件名.路径与文件大小. 代码,shell脚本: lsdir.sh #!/bin/bash # #site: www.jquerycn.cn funct ...

  3. Webpack学习篇

    <深入浅出Webpack>优化篇 01 Webpack 优化可以分为开发优化和输出质量优化两部分,主要要点如下: 优化开发体验,提升开发效率 优化构建速度 优化使用体验 优化输出质量 减少 ...

  4. 并行Louvain社区检测算法

    因为在我最近的科研中需要用到分布式的社区检测(也称为图聚类(graph clustering))算法,专门去查找了相关文献对其进行了学习.下面我们就以这篇论文IPDPS2018的文章[1]为例介绍并行 ...

  5. [BUUCTF]PWN——[V&N2020 公开赛]warmup

    [V&N2020 公开赛]warmup 附件 步骤: 例行检查,64位程序,除了canary,其他保护都开 本地运行一下,看看大概的情况 64位ida载入,从main函数开始看程序 看到程序将 ...

  6. [BUUCTF]PWN——level4

    level4 附件 步骤: 例行检查,32位程序,开启了NX保护 运行一下程序,看看大概的情况 32位ida载入,首先检索程序里的字符串,根据上一步运行看到的字符串进行跳转 输入点在function里 ...

  7. python selenium表单定位

    在Web应用中经常会遇到frame/iframe 表单嵌套页面的应用,WebDriver 只能在一个页面上对元素识别与定位,对于frame/iframe 表单内嵌页面上的元素无法直接定位.这时就需要通 ...

  8. Table.Range保留中间指定的….Range/Middle(Power Query 之 M 语言)

    数据源: "姓名""基数""个人比例""个人缴纳""公司比例""公司缴纳"&qu ...

  9. CF80B Depression 题解

    Content 有一个时针,给定时间为 \(\text{HH}\) 时 \(\text{MM}\) 分,求图中 \(\alpha\) 和 \(\beta\) 角的值. 手画勿喷/kk 数据范围:\(0 ...

  10. python3 迭代器&生成器

    前戏:列表生成式 等于 用列表生成式生成列表.需要将所有数据生成到内存中,占用空间,如果数据太多.生成数据就会耗时较久. 例如需要运行卡顿一下..... 定义一个生成器:定义时不生成任何数据,只有通过 ...