通过对Configuration源代码的分析从而来自定义一个配置数据源

1、用反编译工具来看看AddJsonFile()这个方法究竟干了什么,源代码如下:

        public static IConfigurationBuilder AddJsonFile(this IConfigurationBuilder builder, string path)
{
return builder.AddJsonFile(null, path, false, false);
} public static IConfigurationBuilder AddJsonFile(this IConfigurationBuilder builder, string path, bool optional)
{
return builder.AddJsonFile(null, path, optional, false);
} public static IConfigurationBuilder AddJsonFile(this IConfigurationBuilder builder, string path, bool optional, bool reloadOnChange)
{
return builder.AddJsonFile(null, path, optional, reloadOnChange);
}

首先我们可以看到这3个重载方法,它们都是调用了下面这个方法:

        public static IConfigurationBuilder AddJsonFile(this IConfigurationBuilder builder, IFileProvider provider, string path, bool optional, bool reloadOnChange)
{
if (builder == null)
{
throw new ArgumentNullException("builder");
}
if (string.IsNullOrEmpty(path))
{
throw new ArgumentException(Microsoft.Extensions.Configuration.Json.Resources.Error_InvalidFilePath, "path");
}
return builder.AddJsonFile(delegate (JsonConfigurationSource s) {
s.FileProvider = provider;
s.Path = path;
s.Optional = optional;
s.ReloadOnChange = reloadOnChange;
s.ResolveFileProvider();
});
}

而这个方法呢,又是调用了下面这个方法:

        public static IConfigurationBuilder AddJsonFile(this IConfigurationBuilder builder, Action<JsonConfigurationSource> configureSource)
{
return builder.Add<JsonConfigurationSource>(configureSource);
}

从上面的方法可以看出,类型参数为 JsonConfigurationSource 然后返回IConfigurationBuilder的扩展方法 Add<TSource>(this IConfigurationBuilder builder, Action<TSource> configureSource)

代码如下:

        public static IConfigurationBuilder Add<TSource>(this IConfigurationBuilder builder, Action<TSource> configureSource) where TSource: IConfigurationSource, new()
{
TSource source = Activator.CreateInstance<TSource>();
if (configureSource != null)
{
configureSource(source);
}
return builder.Add(source);
}

能过这个方法,创建 JsonConfigurationSource 的一个实例,最后返回 IConfigurationBuilder  中的 Add(IConfigurationSource source) 方法,接下来我们来看看在这个方法里都干了什么吧

        public IConfigurationBuilder Add(IConfigurationSource source)
{
if (source == null)
{
throw new ArgumentNullException("source");
}
this.Sources.Add(source);
return this;
} public IList<IConfigurationSource> Sources
{
[CompilerGenerated]
get
{
return this.<Sources>k__BackingField;
}
}

通过以上代码我知道了,AddJsonFile()这个方法,创建了一个 JsonConfigurationSource 的一个实例,并将AddJsonFile()中的参数给了这个对象,然后将这个对象存到一个集合中。

2、接下来我们来看看Build()这个方法干了什么:

        public IConfigurationRoot Build()
{
List<IConfigurationProvider> list = new List<IConfigurationProvider>();
using (IEnumerator<IConfigurationSource> enumerator = this.Sources.GetEnumerator())
{
while (enumerator.MoveNext())
{
IConfigurationProvider provider = enumerator.Current.Build(this);
list.Add(provider);
}
}
return new ConfigurationRoot((IList<IConfigurationProvider>) list);
}

这个方法通过遍历你通过AddJsonFile()方法存放进去的 IConfigurationSource 的集合,然后通过调用 IConfigurationSource 的实现类的 Build方法 ,我们来看看这个方法是怎样的:

        public override IConfigurationProvider Build(IConfigurationBuilder builder)
{
base.EnsureDefaults(builder);
return new JsonConfigurationProvider(this);
}

这个方法返回一个 IConfigurationProvider  对象,被赋给了 List<IConfigurationProvider>  集合,接下来我们来看看 JsonConfigurationProvider 这个类里面有什么

        public override void Load(Stream stream)
{
try
{
base.Data = JsonConfigurationFileParser.Parse(stream);
}
catch (JsonReaderException exception)
{
string str = string.Empty;
if (stream.get_CanSeek())
{
stream.Seek((long) 0L, (SeekOrigin) SeekOrigin.Begin);
using (StreamReader reader = new StreamReader(stream))
{
IEnumerable<string> fileContent = ReadLines(reader);
str = RetrieveErrorContext(exception, fileContent);
}
}
throw new FormatException(Microsoft.Extensions.Configuration.Json.Resources.FormatError_JSONParseError((int) exception.LineNumber, str), exception);
}
}

我们看到了这里面有一个Load()方法,这个方法就是用来加载文件中的数据的。

然后我们接着分析上面还没分析的  return new ConfigurationRoot((IList<IConfigurationProvider>) list); 这段代码

这段代码返回了一个 ConfigurationRoot 对象,还将我们所有的 IConfigurationProvider 给传了进行,它想干嘛呢?

没错,上面虽然有Load()方法,但是没人调用啊。 ConfigurationRoot  这个类就是干这个的。

        public ConfigurationRoot(IList<IConfigurationProvider> providers)
{
if (providers == null)
{
throw new ArgumentNullException("providers");
}
this._providers = providers;
foreach (IConfigurationProvider p in providers)
{
p.Load();
ChangeToken.OnChange(delegate {
return p.GetReloadToken();
}, delegate {
this.RaiseChanged();
});
}
}

通过以上代码的分析我们知道了Configuration的底层运行机制

3、那么接下来我们来自定义一个获取自定义的文件的Configuration,能过以上代码我们可以分析出有几个重要的对象

  1、JsonConfigurationSource:最后从这个对象的基类的Data属性中取值

  2、JsonConfigurationProvider:这个里面的Load()方法用来加载文件数据

  3、JsonConfigurationExtensions:这个是用来定义AddJsonFile()方法的

我们规定我们的文件后缀为.custom

CustomConfigurationSource类代码

    public class CustomConfigurationSource : FileConfigurationSource
{
private string path = string.Empty;
public CustomConfigurationSource(string path)
{
this.path = path;
}
public override IConfigurationProvider Build(IConfigurationBuilder builder)
{
return new CustomConfigurationProvider(this.path);
}
}

CustomConfigurationProvider类代码

    public class CustomConfigurationProvider : ConfigurationProvider
{
private string path = string.Empty;
public CustomConfigurationProvider(string path)
{
this.path = path;
}
public override void Load()
{
var lines = System.IO.File.ReadAllLines(this.path);
var dict = new Dictionary<string, string>();
foreach (var item in lines)
{
var items = item.Split("=");
dict.Add(items[], items[]);
}
base.Data = dict;
}
}

CustomConfigurationExtensions类代码 :

    public static class CustomConfigurationExtensions
{
public static IConfigurationBuilder AddCustomFile(this IConfigurationBuilder builder, string path)
{
return builder.Add(new CustomConfigurationSource(path));
}
}

最终调用代码:

        static void Main(string[] args)
{
IConfiguration configuration = new ConfigurationBuilder()
.SetBasePath(Environment.CurrentDirectory)
.AddCustomFile("1.custom")
.Build();;
var info = configuration["host"];
Console.WriteLine(info);
Console.ReadKey();
}

结果如下:

.net core系列之《从源码对Configuration的底层运行机制进行分析》的更多相关文章

  1. .net core系列之《.net core内置IOC容器ServiceCollection》

    一.IOC介绍 IOC:全名(Inversion of Control)-控制反转 IOC意味着我们将对象的创建控制权交给了外部容器,我们不管它是如何创建的,我们只需要知道,当我们想要某个实例时,我们 ...

  2. Asp.Net Core 内置IOC容器的理解

    Asp.Net Core 内置IOC容器的理解 01.使用IOC容器的好处 对接口和实现类由原来的零散式管理,到现在的集中式管理. 对类和接口之间的关系,有多种注入模式(构造函数注入.属性注入等). ...

  3. net core体系-web应用程序-4net core2.0大白话带你入门-8asp.net core 内置DI容器(DependencyInjection,控制翻转)的一点小理解

    asp.net core 内置DI容器的一点小理解   DI容器本质上是一个工厂,负责提供向它请求的类型的实例. .net core内置了一个轻量级的DI容器,方便开发人员面向接口编程和依赖倒置(IO ...

  4. NET Core 3.0 AutoFac替换内置DI的新姿势

    原文:NET Core 3.0 AutoFac替换内置DI的新姿势 .NET Core 3.0 和 以往版本不同,替换AutoFac服务的方式有了一定的变化,在尝试着升级项目的时候出现了一些问题. 原 ...

  5. 浏览器扩展系列————给MSTHML添加内置脚本对象【包括自定义事件】

    原文:浏览器扩展系列----给MSTHML添加内置脚本对象[包括自定义事件] 使用场合: 在程序中使用WebBrowser或相关的控件如:axWebBrowser等.打开本地的html文件时,可以在h ...

  6. 简单讲解Asp.Net Core自带IOC容器ServiceCollection

    一.  理解ServiceCollection之前先要熟悉几个概念:DIP.IOC.DI.Ioc容器: 二.  接下来先简单说一下几个概念问题: 1.DIP(依赖倒置原则):六大设计原则里面一种设计原 ...

  7. ASP.NET Core 2.1 : 十二.内置日志、使用Nlog将日志输出到文件

    应用离不开日志,虽然现在使用VS有强大的调试功能,开发过程中不复杂的情况懒得输出日志了(想起print和echo的有木有),但在一些复杂的过程中以及应用日常运行中的日志还是非常有用. ASP.NET ...

  8. asp.net core 内置DI容器的一点小理解

    DI容器本质上是一个工厂,负责提供向它请求的类型的实例. .net core内置了一个轻量级的DI容器,方便开发人员面向接口编程和依赖倒置(IOC). 具体体现为Micorosoft.Extensio ...

  9. 学习笔记:GLSL Core Tutorial – Vertex Shader(内置变量说明)

    1.每个Vertex Shader都有用户定义的输入属性,例如:位置,法线向量和纹理坐标等.Vertex Shaders也接收一致变量(uniform variables). uniform vari ...

  10. .net core 2.0学习记录(三):内置IOC与DI的使用

    本篇的话介绍下IOC和ID的含义以及如何使用.Net Core中的DI. 一.我是这么理解IOC和DI的: IOC:没有用IOC之前是直接new实例来赋值,使用IOC之后是通过在运行的时候根据配置来实 ...

随机推荐

  1. Win10 VS2015 静态编译Qt5.6.2源码

    由于VS2015需要CRT等拓展组件,因此把内部编写的工具软件以静态发布,固需要编译Qt源码.Qt5.6.2版本,VS2015,Win10 1.安装python,perl,下载jom 2.改文件com ...

  2. 【Qt开发】QDate类

    QDate为开发者提供日期的类,函数也很丰富 常用方法介绍 1.QDate addDays(qint64 ndays) const 当前日期添加n天,n可以为负 2.QDate addMonths(i ...

  3. emacs26.1 ppa

    sudo add-apt-repository ppa:kelleyk/emacssudo apt updatesudo apt install emacs26

  4. goahead3.6.3就基本使用(后台上传信息到html页面),高手请忽略

    声明:这里面的代码均为网上找的然后有小小的改动,并非原创.但文章为原创 一..编译 1.1,.下载:进入http://embedthis.com/goahead/下载goahead3.6.3(2017 ...

  5. (转载)GRASP职责分配原则

    GRASP (职责分配原则) 要学习设计模式,有些基础知识是我们必须要先知道的,设计模式是关于类和对象的一种高效.灵活的使用方式,也就是说,必须先有类和对象,才能有设计模式的用武之地,否则一切都是空谈 ...

  6. BST转换成有序链表

    把二元查找树转变成排序的双向链表(树)题目:输入一棵二元查找树,将该二元查找树转换成一个排序的双向链表.要求不能创建任何新的结点,只调整指针的指向. struct BSTreeNode{ int va ...

  7. Linq to SharePoint与权限提升(转)

    转自http://www.cnblogs.com/kaneboy/archive/2012/01/25/2437086.html SharePoint 2010支持Linq to SharePoint ...

  8. 用Ajax爬取今日头条图片集

    Ajax原理   在用requests抓取页面时,得到的结果可能和浏览器中看到的不一样:在浏览器中可以正常显示的页面数据,但用requests得到的结果并没有.这是因为requests获取的都是原始 ...

  9. JS日期、月份的加减

    JS日期.月份的加减 需要注意的是返回的月份是从0开始计算的,也就是说返回的月份要比实际月份少一个月,因此要相应的加上1 // 日期,在原有日期基础上,增加days天数,默认增加1天 function ...

  10. LINQ-Where子句与select子句

    1.Where子句 where子句的作用就是筛选元素,除了开始喝结束位置,where子句几乎可以出现在LINQ表达式的任意位置.一个LINQ表达式中可以有Where子句,也可以没有:可以有一个,可以有 ...