.net core 自定义配置系统

前言

今天来简单写一个自定义的配置系统,能自己手写一些稍微底层的东西,对之后的使用和学习都有帮助。在.net core web api 项目中,默认有一个配置文件叫appsetting.json,可以在这个文件中定义一些配置项,需要使用的时候直接使用IConfiguration即可,那么今天咱们就来实现一下这个东西的内部逻辑。

实战准备

首先还是新建一个控制台项目。然后新建一个名为appsetting.json的配置文件。内容如下,这里我就写的很简单了,因为本篇文章主要是学习如何自定义配置系统,如果想要复杂那就可以在基础之上在完善。

{
  "Address": "127.0.0.1",
  "Port": "80"
}

由于我们这个配置系统是默认读取 appsetting.json这个文件。所以主要依赖的是FileConfigurationProvider这个类。
首先我们新建一个LmConfigProvider类,这个类继承自FileConfigurationProvider,然后需要重写Load方法。

using Microsoft.Extensions.Configuration;
using Newtonsoft.Json.Linq;

namespace 自定义配置系统
{
    public class LmConfigProvider:FileConfigurationProvider
    {
        public LmConfigProvider(LmConfigSource src):base(src)
        {
        }
        public override void Load(Stream stream)
        {
            var data = new Dictionary<string, string?>();
            var reader = new StreamReader(stream);
            var fileInfo =  reader.ReadToEnd();
            JObject jobject = (JObject)JToken.Parse(fileInfo);
            foreach (var item in jobject)
            {
                data.Add(item.Key, item.Value.ToString());
            }
            this.Data = data;
        }
    }
}

Load方法,我的实现比较简单,默认传入的是一个Stream流,所以直接读取这个Json文件就可以,如果文件内容里有嵌套结构,则需要进行扁平化处理,比如A:B,A:C,如果有数组则还需要加上数组下标,比如,A:D:0,A:D:1,表示这样一种结构。

{
    "A":{
            "B":"hello",
            "C":"world",
            "D":["abc","def"]
        }
}

那么A:D:0=abc,A:D:1=def,想深入了解的同学可以继续研究。我这里就简单把JSON转成JObject对象,然后遍历KeyValue然后添加到字典中,最后给ConfigurationProviderData属性赋值。
然后添加一个类LmConfigSource

using Microsoft.Extensions.Configuration;
namespace 自定义配置系统
{
    public class LmConfigSource : FileConfigurationSource
    {
        public override IConfigurationProvider Build(IConfigurationBuilder builder)
        {
            //加载默认项
            EnsureDefaults(builder);
            return new LmConfigProvider(this);
        }
    }
}

这个类就比较简单,重写一下IConfigurationProvider方法,把我们的实现类LmConfigProvider直接返回就行。
然后我们的基本工作就算做完了,就添加了两个类。然后在主函数Main里添加如下代码。

var configBuilder = new ConfigurationBuilder();
configBuilder.Add(new LmConfigSource() { Path="appsetting.json" });
var config = configBuilder.Build();
string addr = config["Address"];
string port = config.GetSection("Port").Value;
Console.WriteLine($"Address={addr}");
Console.WriteLine($"Port={port}");

运行一下,就可以看到结果了。
在这里插入图片描述

到这里其实应该就可以结束了,但是呢,在实际使用过程中,是不会添加那么一大段读取配置文件的方法的,所以我们可以新建一个扩展方法来自动帮我们完成这些配置工作。
新增ConfigExtension类。实现如下。

using Microsoft.Extensions.Configuration;
namespace 自定义配置系统
{
    public static class ConfigExtension
    {
        public static IConfigurationRoot UseLmConfig(this IConfigurationBuilder builder)
        {
            var configBuilder = new ConfigurationBuilder();
            configBuilder.Add(new LmConfigSource() { Path="appsetting.json" });
            return configBuilder.Build();
        }
    }
}

那么我们在使用过程中就可以省略成。

    var builder = new ConfigurationBuilder();
    builder.UseLmConfig();

如果有Startup类的话,在Configure方法下直接添加app.UseLmConfig();其实就可以了。 那么我在控制台里使用依赖注入的方式来验证一下。 还是在添加一个用来测试读取配置的类。然后注入IConfiguration

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;

namespace 自定义配置系统
{
    public class TestController
    {
        private readonly IConfiguration config;
        public TestController(IConfiguration configuration) 
        { 
            config=configuration; 
        }
        public void Test()
        {
            string addr = config["Address"];
            string port = config.GetSection("Port").Value;
            Console.WriteLine($"Address={addr}");
            Console.WriteLine($"Port={port}");
        }
    }
}

然后主函数添加如下代码。

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;

namespace 自定义配置系统
{
    internal class Program
    {
        static void Main(string[] args)
        {
            ServiceCollection services = new ServiceCollection();
            services.AddScoped<IConfiguration>(serviceProvider =>
                new ConfigurationBuilder().UseLmConfig()
            );
            services.AddScoped<TestController>();
            services.AddScoped<LmConfigProvider>();
            services.AddScoped<LmConfigSource>();

            using (var sp = services.BuildServiceProvider())
            {
                var test = sp.GetRequiredService<TestController>();
                test.Test();
            }

            Console.ReadKey();
        }
    }
}

注入IConfiguration的时候,需要加上我们的扩展方法,因为已经不是用原来的方法了。
可以看到依然读取出来了。

在这里插入图片描述

结语

通过自定义实现配置系统,可以了解到配置系统的原理,以后在使用第三方的配置系统比如ApolloNacos等,也会更加的得心应手。

Study hard and make progress every day.

欢迎关注下方微信公众号,一起学习,一起娱乐,一起进步,点击卡片可以查看公众号二维码哟。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

李公子lm

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值