.NET 控制台后台程序实践细节总结

13 阅读2分钟

使用 Host 通用主机和直接创建手动创建 ServiceCollection 构建 DI 容器的区别

主要区别对比

特性ServiceCollection Host
功能完整性基础 DI 容器完整应用框架(DI + 配置 + 日志 + 生命周期)
配置系统需手动添加默认集成(自动加载 appsettings.json)
日志系统需手动配置默认集成(Console/Debug/EventLog 等)
后台服务不支持支持 IHostedService/BackgroundService
生命周期管理手动管理作用域自动管理应用生命周期
环境区分需手动实现内置开发/生产环境支持
扩展性有限通过 Configure* 方法高度可扩展

ServiceCollection 示例

// 手动管理所有组件
var services = new ServiceCollection();

// 注入日志操作
services.AddLogging(builder => builder.AddNLog("NLog.dev.config"));

// 注入配置
ConfigurationBuilder builder = new();
builder.AddAppJsonFile("appsettings.json", optional: false, reloadOnChange: true);
builder.AddCommandLine(args);
IConfigurationRoot root = builder.Build();

// 绑定配置
services.AddOptions().Configure<AppSettings>(root.Bind);
services.AddOptions().Configure<DatabaseConfig>(root.GetSection("database").Bind);

// 注册服务
services.AddSingleton<IMyService, MyService>();
services.AddTransient<Worker>();

/* 其他服务注入 */

// 手动构建容器
using var provider = services.BuildServiceProvider();

// 手动创建作用域
using var scope = provider.CreateScope();
var worker = scope.ServiceProvider.GetRequiredService<Worker>();
worker.Execute();

Host 示例

var host = Host.CreateDefaultBuilder(args)
  //.ConfigureAppConfiguration((context, config) =>{ /* 默认加载 appsetting.json 和对应的环境配置 */})
  //.ConfigureLogging((context, logging) => {/* 使用 NLog 无需再设置 Logging */})
  .ConfigureServices((context, services) =>
  {
      // 绑定配置信息
      services.Configure<AppSettings>(context.Configuration);
      services.Configure<DatabaseConfig>(context.Configuration.GetRequiredSection("database")); 
      // 注册后台服务
      services.AddHostedService<WorkerService>();
      services.AddScoped<MyTaskService>();      
      /* 其他服务注入 */
  })
  // 使用 NLog 日志并替换自带的日志,需要安装 NLog.Extensions.Hosting
  .UseNLog(new NLogProviderOptions() { ReplaceLoggerFactory = true })
  .UseConsoleLifetime()
  .Build();

// 注册关闭事件
//var appLifetime = host.Services.GetRequiredService<IHostApplicationLifetime>();
//appLifetime.ApplicationStopping.Register(() => commandManager.Dispose());

// 运行主机
await host.RunAsync();

NLog 实践细节

日志作用域的传递性

  • logger 会自动继承上下文,调用位置的参数会传递到内部调用的对象
  • 自动处理异步方法的传递,需要使用 await ,日志作用域通过 AsyncLocal<T> 会自动在异步调用链中流动
  • Parallel.For、new Thread()、Task.Run 无法自动传递作用域
  • 使用作用域可以优化日志文件的管理和层次结构
  • 示例
    public class HandlerTaskService(PluginTaskService pluginTaskService, ILogger<HandlerTaskService> logger)
    {
        private readonly CancellationTokenSource cts = new();
        public async Task Start(HandlerConfig handler, CancellationToken token)
        {
            // 向内部的 Plugin 传递日志作用域参数 HandlerName
            using var scope = logger.BeginScope("Handler:{HandlerName}", handler.Name);
            logger.LogInformation("正在启动 [{name}] 处理程序...", handler.Name);
            var tasks = handler.Plugins.Select(plugin => pluginTaskService.Start(plugin, handler, cts.Token));
            await Task.WhenAll(tasks); // 必须 await
        }
    }
    
    public class PluginTaskService(IPluginServiceFactory factory, ILogger<PluginTaskService> logger)
    {
        public async Task Start(PluginConfig plugin, HandlerConfig handler, CancellationToken _token)
        {
            // 此时 logger 包含调用该方法时的日志作用域上下文,同时向内部传递 PluginName 参数
            using var scope = logger.BeginScope("Plugin:{PluginName}", plugin.Name);
            logger.LogInformation("启动插件 [{name}] ...");
            await Task.Delay(1000, _token);
        }
    }
    

配置文件

<!-- NLog.config -->
<?xml version="1.0" encoding="utf-8" ?>
<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <!-- 获取 logger 设置的 category , ${logger:shortName=true} 不含命名空间的类名称 -->
    <variable name="category" value="${logger}" />
    <!-- 获取 scope 设置的 HandlerName 参数, :whenEmpty=Shared 提供了空参数时的默认值-->
    <variable name="handler" value="${scopeproperty:item=HandlerName:whenEmpty=Shared}" />
    <!-- 使用 HandlerName 参数作为目录结构的组成部分 -->
    <variable name="dir" value="${basedir}/logs/${handler}" />

    <variable name="layout" value="${longdate} ${uppercase:${level}:padding=-5}: [${category}] ${message}" />
    <variable name="consoleLayout" value="${longdate} ${uppercase:${level}:padding=-5}: [${handler}][${category}] ${message}" />

    <targets async="true">
            <target name="console" xsi:type="ColoredConsole" layout="${consoleLayout}"/>
            <target name="file" xsi:type="File" layout="${layout}"
                            fileName="${dir}/${shortdate}.log"
                            concurrentWrites="true"
                            keepFileOpen="true"
                            archiveFileName="${dir}/archives/${shortdate}_{#}.log"
                            archiveAboveSize="10485760"
                            maxArchiveFiles="10"
                            archiveNumbering="Rolling"
                            maxArchiveDays="30" />
    </targets>
</nlog>



<!-- NLog.dev.config -->
<?xml version="1.0" encoding="utf-8" ?>
<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
	<include file="NLog.config" />
	<rules>
		<logger name="*" minlevel="Debug" writeTo="console,file" />
		<!--<logger name="*" minlevel="Error" writeTo="errfile"/>-->
		<!--<logger name="*" minlevel="Warning" writeTo="warnfile" final="true" />-->
	</rules>
</nlog>