zoukankan      html  css  js  c++  java
  • 将配置绑定为对象

     将配置绑定为对象

    我们在《读取配置信息》通过实例的形式演示了如何利用Options模型以依赖注入的方式直接获取由指定配置节绑定生成的Options对象,我们再次回顾一下当初我们编写的程序。如下面的代码片段所示,基于Options模型的配置绑定的编程基本采用这样的模式:先后调用ServiceCollection的扩展方法AddOption和Configure注册Options模型相关的服务并完成Options类型与指定配置节之间的映射,然后利用由此生成ServiceProvider获得一个类型为IOptions<TOptions>的服务示例,后者的Value就是配置绑定生成的Options对象。

       1: FormatSettings settings = new ServiceCollection()
       2:     .AddOptions()
       3:     .Configure<FormatSettings>(configuration)
       4:     .BuildServiceProvider()
       5:     .GetService<IOptions<FormatSettings>>()
       6:     .Value;


    一、IOptions <TOptions>

    由于Options模型的编程仅仅涉及到上述几个方法的调用,所以只要搞清楚这几个方法背后的实现逻辑,我们也就彻底了解了Options模型的实现原理。首先当我们调用ServiceCollection的扩展方法时,实际上仅仅是按照如下的方式注册了一个针对IOptions <TOptions>接口类型的服务而已。服务接口IOptions<TOptions>仅仅定义了一个只读属性Value,该属性返回的正是绑定了指定配置数据的Options对象。

       1: public static class ServiceCollectionExtensions
       2: {
       3:     public static IServiceCollection AddOptions(this IServiceCollection services)
       4:     {
       5:         return services.AddSingleton(typeof(IOptions<>), typeof(OptionsManager<>));
       6:     }
       7: }
       8:  
       9: public interface IOptions<TOptions> where TOptions:class, new()
      10: {
      11:     TOptions Value { get; }
      12: }

    通过上面的给出的代码片段我们不难看出,AddOptions方法实际上是以Singleton模式注册了一个类型为OptionsManager<TOptions>的服务,如下所示的代码片段基本反映了该类型的实现逻辑。如下面的代码片段所示,OptionsManager<TOptions>的只读属性Value返回的Options对象是以“延迟加载(Lazy Loading)”的形式被提供。Options对象创建的逻辑也很简单,我们直接调用其默认构造函数创建一个空的Options对象,然后将其递交给在构造函数中指定的一系列IConfigureOptions<TOptions>进行设置,配置绑定就这这个过程中完成。

       1: public class OptionsManager<TOptions> : IOptions<TOptions> where TOptions : class, new()
       2: {
       3:     private Lazy<TOptions> optionsAccessor;
       4:     public OptionsManager(IEnumerable<IConfigureOptions<TOptions>> setups)
       5:     {
       6:         optionsAccessor = new Lazy<TOptions>(() =>
       7:         {
       8:             TOptions options = new TOptions();
       9:             setups.ForEach(it => it.Configure(options));
      10:             return options;
      11:         });
      12:     }
      13:     public TOptions Value
      14:     {
      15:         get { return optionsAccessor.Value; }
      16:     }
      17: }


    二、IConfigureOptions<TOptions>

    IConfigureOptions<TOptions>接口抽象了针对Options对象的配置行为,这个行为体现在定义其中的Configure方法。ConfigureOptions<TOptions>实现了这个接口,它采用在构造函数提供的Action<TOptions>完成对Options对象的配置。

       1: public interface IConfigureOptions<TOptions> where TOptions : class, new()
       2: {
       3:     void Configure(TOptions options);
       4: }
       5:  
       6: public class ConfigureOptions<TOptions> : IConfigureOptions<TOptions> where TOptions : class, new()
       7: {
       8:     public Action<TOptions> Action { get; private set; }
       9:     public ConfigureOptions(Action<TOptions> action)
      10:     {
      11:         this.Action = action;
      12:     }
      13:     public void Configure(TOptions options)
      14:     {
      15:         this.Action(options);
      16:     }
      17: }

    针对Options对象的配置绑定工作实现在一个名为ConfigureFromConfigurationOptions<TOptions>的类中。如下面的代码片段所示,这个类型直接继承ConfigureOptions<TOptions>,在构造函数中指定的Configuration对象承载了最终需要绑定到Options对象上的配置数据,它直接调用Configuration对象的扩展方法Bind完成了针对Options对象的配置绑定。

       1: public class ConfigureFromConfigurationOptions<TOptions>: ConfigureOptions<TOptions> where TOptions : class, new()
       2: {
       3:     public ConfigureFromConfigurationOptions(IConfiguration configuration) : base(options => configuration.Bind(options))
       4:     { }
       5: }

    在Options模型中,ConfigureFromConfigurationOptions<TOptions>对象通过扩展方法Configure方法被注册到指定的ServiceCollection对象中。如下面的代码片段所示,Configure方法直接利用作为参数传入的Configuration对象创建一个ConfigureFromConfigurationOptions<TOptions>对象,并将这个对象注册到ServiceCollection之中。

       1: public static class ServiceCollectionExtensions
       2: {
       3:     public static IServiceCollection Configure<TOptions>(this IServiceCollection services, IConfiguration configuration) 
       4:     where TOptions : class, new()
       5:     {
       6:         return services.AddInstance<IConfigureOptions<TOptions>>(new ConfigureFromConfigurationOptions<TOptions>(configuration));
       7:     }
       8: }


    三、Options对象的提供

    9整个Options模型以两个注册到ServiceCollection的服务为核心,这两个服务对应的服务接口分别是IOptions <TOptions>和IConfigureOptions<TOptions>,前者直接提供最终绑定了配置数据的Options对象,后者则在Options对象返回之前对它实施相应的初始化工作。这个两个服务分别通过扩展方法AddOptions和Configure方法注册到指定的ServiceCollection之中,服务的真实类型分别是OptionsManager<TOptions>和ConfigureFromConfigurationOptions<TOptions>,后者派生于ConfigureOptions<TOptions>。右图所示的UML体现了Options模型中涉及的这些接口和类型之间的关系。

    对于一个包含服务注册描述信息的ServiceCollection,当我们分别调用其扩展方法AddOptions和Configure完成了相应的服务注册之后,我们就可以利用由它生成的ServiceProvider对象来提供针对接口类型IOptions <TOptions>的服务实例,并通过后者的只读属性Value得到配置绑定生成的Options对象。

    ServiceProvider提供的这个服务实例自然是一个OptionsManager<TOptions>对象,当ServiceProvider调用构造函数对它进行实例化的时候,我们注册的ConfigureFromConfigurationOptions<TOptions>对象会以构造器注入的形式作为参数。在构造函数执行过程中,一个空的Options对象先被创建出来后会作为参数调用ConfigureFromConfigurationOptions<TOptions>的Configure方法,后者将在预先指定的Configuration对象绑定到这个Options对象之上。

    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):配置的同步[下篇]

    作者:蒋金楠 
    微信公众账号:Artech1981
  • 相关阅读:
    表:t_category
    使用 Javascript 代码,增加 HTML 新元素(节点)
    Kind Editor 笔记
    [ 摘 ] 对 js 运算符 “||” 和 “&&” 的总结
    Oracle 的分页查询,myBatis 配置文件
    ExtJS 的工具条及菜单
    发一个 ExtJS 4.2.0 的页面源码(规则比对公式 的设置页面)
    MyBatis 中的 set 元素用法(MyBatis 3.1.1)
    MySQL 存储过程,一个 4 层的游标,用于规则比对,不知道性能如何。
    jmeter从文件中读取参数
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/5437597.html
Copyright © 2011-2022 走看看