前言

之前就写过 Asp.net core 学习笔记 ( Configuration 配置 ) 只是有点乱, 这篇作为整理版.

项目中会有许许多多的 Config 要设定. 比较好的管理方式是把它们放到 json file 里. 这样想修改时就不需要改动源码, 改 json file 就行了.

ASP.NET Core 提供了一套管理 Config 的方式. 这篇主要就是介绍这个.

参考:

docs – Configuration in ASP.NET Core

appsetting.json

appsetting.json 就是 ASP.NET Core 的 config file, 我们可以把所有模块用到的 config 都写在里面.

真实情况大概长这样

For 下面的测试, 我做一个简单的就好了

{
"MyConfig": {
"Child1": {
"Key": "Value",
"Secret": "secret"
},
"Child2": {
"Key": "Value"
}
}
}

另外, 它还支持多环境 config

appsettings.json 是抽象, appsettings.Development.json 是具体, 具体可以 override 和 extend 抽象.

"MyConfig": {
"Child2": {
"Key": "New Value"
},
"Child3": {
"Key": "New Value"
}
}

注: Child2 的 Key 是 override, MyConfig.Child3 是 extend, 没办法 override 整个 MyConfig 对象的.

User Secrets

想深入了解请看这篇 ASP.NET Core – User Secret & Azure Key Vault .

有一些 config 比较敏感, 比如密码. ASP.NET Core 提供了一个叫 User Secrets 的方案来解决这个问题.

上面例子中, MyConfig.Child.Secret 是敏感数据. 不应该直接把 value 写到 json file 里面, 必须使用 User Secrets.

dotnet user-secrets init
dotnet user-secrets set "MyConfig:Child1:Secret" "password"

注意, 它的分隔符是分号 ":" 而不是点 "." 哦, 如果是 Array 就写号码, 比如: MyConfig:Array:0

这个 password 会被存到另一个 local file, git checkin 只会把 appsetting.json checkin, User Secrets local file 则不会, 所以密码只会留在电脑中.

项目发布时, 则会通过 Azure KeyVault 来充当这个 User Secrets, 所以在 Web Server appsetting.json 依然不会有任何敏感数据.

好了, 这样我们的 config 定义就完成了. 接下来看看如何在项目中获取这些 config.

Get Config Value

GetValue

program.cs

var builder = WebApplication.CreateBuilder(args);
var configValue1 = builder.Configuration.GetValue<string>("MyConfig:Child1:Key");
var configValue2 = builder.Configuration.GetValue<string>("MyConfig:Child1:Secret");

.NET 6 以前, 想在 program.cs 或者 config 是很难的, 但是现在很简单直观了.

CreateBuilder 会把 appsettings.Development.json, appsettings.json, User Secrets 弄好好.

通过 builder.Configuration.GetValue("path") 就可以获取到任何 value 了.

注意, path 的分隔符是分号 ":" 而不是点 "." 哦.

GetValue 一定要声明类型, 如果不清楚类型可以这样获取

var value = builder.Configuration["MyConfig:Child1:Key"];

value 的值一定是 string?,

如果是 null 那么就是 empty string.

如果是 boolean 那么就是 "False" or "True"

如果是 object 或 array 那么是 null

注: 最好能清楚 config 结构和类型, 不然会很乱的

GetSection

获取整个对象.

public class Child
{
public string Key { get; set; } = "";
} var childObject = builder.Configuration.GetSection("MyConfig:Child1").Get<Child>();

Section != Child 对象哦, 所以要记得 .Get() 才能获取到 Child 对象.

在 Console 创建和获取 Configuration

虽然 WebApplication.CreateBuilder 已经帮我们弄美美了, 但为了能理解多一点底层, 我们也看看 Console 的版本吧

创建项目和安装各做 package

dotnet new console -o SecretConsole

dotnet add package Microsoft.Extensions.Configuration
dotnet add package Microsoft.Extensions.Configuration.FileExtensions
dotnet add package Microsoft.Extensions.Configuration.Json
dotnet add package Microsoft.Extensions.Configuration.UserSecrets
dotnet add package Microsoft.Extensions.Configuration.Binder

program.cs

using System.Reflection;
using Microsoft.Extensions.Configuration; var environment = Environment.GetEnvironmentVariable("ASPNETCORE_ENVIRONMENT"); // Development // 创建 ConfigBuilder
var configurationBuilder = new ConfigurationManager()
.SetBasePath(Directory.GetCurrentDirectory())
.AddJsonFile("appsettings.json", optional: false, reloadOnChange: true)
.AddJsonFile($"appsettings.{environment ?? ""}.json", optional: true, reloadOnChange: true); // Development 情况下用本地的 UserSecrets
if (environment == "Development")
{
configurationBuilder = configurationBuilder.AddUserSecrets(Assembly.GetExecutingAssembly());
// configurationBuilder = configurationBuilder.AddUserSecrets<Program>(); // 或者用 Program class 也是一样的
} // 到这里就和 WebApplication.CreateBuilder 的 builder.Configuration 同样用法了
var configure = configurationBuilder.Build();
var value = configure.GetValue<string>("Account:Password");
Console.Write(value); // my password

通过 DI 获取 Configuration

上面 program.cs 是通过 builder 获取到 configuration. 想在 Razor Pages, Controllers, Services 获取到 Configuration 就需要通过 DI

public class IndexModel : PageModel
{
public IndexModel(
IConfiguration configuration
)
{
var value = configuration.GetValue<string>("MyConfig:Child1:Key");
} public void OnGet()
{ }
}

注入 IConfiguration 就可以了.

Options

封装的模块通常不会直接通过 appsetting 获取 configuration (唯一例外的是 Log).

绝大部分的模块会通过 Options 来管理 "Config".

这些就是 options

我们先看看 Options 的玩法, 之后在看它如何和 configuration 一起工作.

Service Module

假设想封装一个服务

service class

public class MyService
{
public string GetComputedValue()
{
return "value";
}
}

provide

builder.Services.AddScoped<MyService>();

inject

public IndexModel(
MyService myService
)
{
var result = myService.GetComputedValue();
}

升级为模块

public static class IServiceCollectionExtensions
{
public static void AddMyModule(this IServiceCollection services)
{
services.AddScoped<MyService>();
}
}

provide

builder.Services.AddMyModule();

Service Options

这时, 想加入一些 config, 在 provide 的时候设定.

public class MyServiceOptions
{
public string Value { get; set; } = "";
}

provider

builder.Services.AddMyModule(options =>
{
options.Value = "my value";
});

module

public static class IServiceCollectionExtensions
{
public static void AddMyModule(this IServiceCollection services, Action<MyServiceOptions> optionsBuilder)
{
var options = new MyServiceOptions();
optionsBuilder(options);
var value = options.Value; // my value
services.AddScoped<MyService>();
}
}

这时问题来了, MyService 和 MyServiceOptions 怎样关联起来呢?

既然是用 DI, MyService 依赖 MyServiceOptions, 那么显然 MyServiceOptions 也必须 provide, 这样才能被 MyService 注入.

ASP.NET Core 提供了一个 services.Configure 接口, 让我们 provide 这个 options.

    public static void AddMyModule(this IServiceCollection services, Action<MyServiceOptions> optionsBuilder)
{
services.Configure<MyServiceOptions>(optionsBuilder);
services.AddScoped<MyService>();
}

Configure 是可以 call multiple times 的哦

好, provide 没有问题了, 那怎么注入?

通过 IOptions<MyServiceOptions> 注入.

public class MyService
{
public MyService(
IOptions<MyServiceOptions> myServiceOptions
)
{
var value = myServiceOptions.Value;
}
public string GetComputedValue()
{
return "value";
}
}

Options Work with Configuration

上面 options 是通过 optionsBuilder 来设置的. 那怎样让它和 appsetting 挂钩呢?

方法 1, 把 section 丢进去.

builder.Services.Configure<MyServiceOptions>(builder.Configuration.GetSection("MyServiceOptions"));

方法 2, 挨个挨个 set

builder.Services.Configure<MyServiceOptions>(options =>
{
options.Value = builder.Configuration.GetValue<string>("MyServiceOptions:Value");
});

这样就行了.

Named Options

我是在玩 Identity External Login 时发现的. GoogleOptions

[FromServices] IOptionsSnapshot<GoogleOptions> googleOptionsAccessor

var googleOptions = googleOptionsAccessor.Get("Google").Scope.ToList();

参考: Docs – Named options support using IConfigureNamedOptions

public static void Main(string[] args)
{
var serviceCollection = new ServiceCollection();
// 做 2 次 Configure<ServiceOptions> 但是用放不同的名字
serviceCollection.Configure<ServiceOptions>("Option1", options => options.Name = "Derrick");
serviceCollection.Configure<ServiceOptions>("Option2", options => options.Name = "Alex"); var serviceProvider = serviceCollection.BuildServiceProvider();
// 获取 options accessor 手法一样
var serviceOptionsAccessor = serviceProvider.GetRequiredService<IOptionsSnapshot<ServiceOptions>>();
// 不同名字可以从 accessor 里拿到不同的 options
Console.WriteLine(serviceOptionsAccessor.Get("Option1").Name); // Derrick
Console.WriteLine(serviceOptionsAccessor.Get("Option2").Name); // Alex
}

看注释理解

Optional Options

假如我忘了在 program.cs register ServiceOptions。

// builder.Services.Configure<ServiceOptions>(options => options.Name = "test");

但我缺尝试去 inject 它

public IndexModel(
IOptionsSnapshot<ServiceOptions> serviceOptionsAccessor
)
{
var serviceOptions = serviceOptionsAccessor.Value;
}

它是不会报错了。它会 new ServiceOptions() 作为 default value。

如果你需要识别出是否有提供 ServiceOptions 会比较麻烦。

参考: Stack Overflow – How to make an IOptions section optional in .NET Core?

在 options 里多加一个 property 来表示。

IOptionsSnapshot vs IOptions vs IOptionsMonitor

上面给的例子是用 IOptions 来注入. 它有个缺点. 就是当 appsetting 修改了以后, 需要重启 app 才能 update.

有时候这个是预期的效果, 但有时候会希望马上更新. 于是有了另外 2 个 IOptions 变种.

参考:

IOptions、IOptionsMonitor、IOptionsSnapshot的区别

ASP.NET Core笔记(4) - 选项模式

它们之间主要是生命周期不同.

IOptions 算是单列, optionsBuilder 只运行一次. 一直到 application 重启,

IOptionsSnapshot 的生命周期是 scope (per request), 它把声明周期从 app 缩小到每个 request.

每一次新的请求就会重跑 optionBuilder 拿到新的 Options 值, 需要注意的是 snapshot 的周期是 scope 也意味着它不能用在单列的 service 哦.

IOptionsMonitor 可以用在单列也可以不需要重启 app, 因为它获取的是 current value. 也就是每一次都拿最新的, 甚至在同一个 request 里面.

我个人的用法是尽可能就用 snapshot 然后少用单列 service.

ASP.NET Core – Configuration & Options的更多相关文章

  1. 理解ASP.NET Core - 选项(Options)

    注:本文隶属于<理解ASP.NET Core>系列文章,请查看置顶博客或点击此处查看全文目录 Options绑定 上期我们已经聊过了配置(IConfiguration),今天我们来聊一聊O ...

  2. asp.net core选项Options模块的笔记

    这篇博客是写给自己看的.已经不止一次看到AddOptions的出现,不管是在.net core源码还是别人的框架里面,都充斥着AddOptions.于是自己大概研究了下,没有深入,因为,我的功力还是不 ...

  3. ASP.Net Core Configuration 理解与源码分析

    Configuration 在ASP.NET Core开发过程中起着很重要的作用,这篇博客主要是理解configuration的来源,以及各种不同类型的configuration source是如何被 ...

  4. ASP.NET Core 中文文档 第二章 指南(4.5)使用 SQL Server LocalDB

    原文:Working with SQL Server LocalDB 作者:Rick Anderson 翻译: 魏美娟(初见) 校对: 孟帅洋(书缘).张硕(Apple).许登洋(Seay) Appl ...

  5. Working with Data » 使用Visual Studio开发ASP.NET Core MVC and Entity Framework Core初学者教程

    原文地址:https://docs.asp.net/en/latest/data/ef-mvc/intro.html The Contoso University sample web applica ...

  6. Prerender Application Level Middleware - ASP.NET Core Middleware

    In the previous post Use Prerender to improve AngularJS SEO, I have explained different solutions at ...

  7. 5. abp集成asp.net core

    一.前言 参照前篇<4. abp中的asp.net core模块剖析>,首先放张图,这也是asp.net core框架上MVC模块的扩展点 二.abp的mvc对象 AbpAspNetCor ...

  8. ASP.NET Core - 选型系统之选型配置

    1. 选项 前面讲完了.NET Core 下的配置系统,我们可以通过 IConfiguration 服务从各种来源的配置中读取到配置信息,但是每次要用的时候都通过 Iconfiguration 读取配 ...

  9. ASP.NET Core - 选项系统之选项验证

      就像 Web Api 接口可以对入参进行验证,避免用户传入非法的或者不符合我们预期的参数一样,选项也可以对配置源的内容进行验证,避免配置中的值与选项类中的属性不对应或者不满足预期,毕竟大部分配置都 ...

  10. ASP.NET Core - 选项系统之源码介绍

    .NET Core 选项系统的主要实现在 Microsoft.Extensions.Options 和 Microsoft.Extensions.Options.ConfigurationExtens ...

随机推荐

  1. 基于vsftpd搭建项目文件服务器

    vsftpd 是"very secure FTP daemon"的缩写,安全性是它的一个最大的特点.vsftpd 是一个 UNIX 类操作系统上运行的服务器的名字,它可以运行在诸如 ...

  2. 微信小程序热门选题

    一.大体实现思路 微信小程序,现在是非常热门的,基于微信生态开发的.现在很多计算机毕业的同学,都会选择微信小程序作为毕业设计 小程序端通常都是展示数据给用户去看的,大多数情况下,这些数据不是写死的,而 ...

  3. [oeasy]python0095_乔布斯求职_雅达利_atari_breakout_打砖块_布什内尔_游戏机_Jobs

    编码进化 回忆上次内容 上次 我们回顾了 电子游戏的历史 从 电子游戏鼻祖 双人网球 到 视频游戏 PingPong 再到 街机游戏 Pong 雅达利 公司 来了 嬉皮士 捣乱? 布什内尔 会如何 应 ...

  4. Charles抓包配置、常见问题和解决方法

    1.下载安装charles,官方下载地址:https://www.charlesproxy.com/download/ 如图,下载所对应系统需要的相应版本即可. 2.解压Charles包,双击Char ...

  5. Gradle的安装和创建java项目(idea)

    安装 Gradle下载地址:http://services.gradle.org/distributions/ 下载后解压. 解压后的目录结结构如下: 新增环境变量 在path环境变量中添加以下内容: ...

  6. app专项测试:测试内容

    app专项测试:测试内容 除了app的UI功能测试,平时听说比较多的就是app专项测试了, app专项测试主要包含以下内容: 1,流量测试 :app静态测试(耗时.流量.内存.图片大小) 2,弱网测试 ...

  7. Jmeter函数助手29-dateTimeConvert

    dateTimeConvert函数用于将源格式进行目标格式的转换. 格式化时间:传入时间参数,此处格式需要与源时间格式一致 源时间格式:传入参数的时间格式 目标时间格式:想要转换成的格式 1.将源格式 ...

  8. 关于spring boot中mapper注入到service时IDEA报错的解决办法

    虽然这个错误不影响正常运行但是作为强迫症患者看着实属难受,经过在论坛查看资料学习到以下两种解决方法,可以供大家参考以下,如有什么错误的地方还希望各位大佬指定一下. 1.在注解@Autowired后增加 ...

  9. ByteHouse高性能向量检索实践——“以图搜图”

    更多技术交流.求职机会,欢迎关注字节跳动数据平台微信公众号,回复[1]进入官方交流群   随着 LLM 技术的发展,向量检索与向量数据库也受到业界持续关注,它们能够为LLM提供外置记忆单元,通过提供与 ...

  10. 8、IDEA集成Git

    8.1.配置Git忽略文件 8.1.1.忽略文件的原因 在使用 IDE 工具时,会自动生成一些和项目源码无关的文件,所以可以让 Git 忽略这些文件. 此外,把这些无关文件忽略掉,还能够屏蔽不同 ID ...