ASP.NET Core的配置(3):将配置绑定为对象[下篇]
我们在《读取配置信息》通过实例的形式演示了如何利用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对象的提供
整个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对象之上。