我們都知道,在 ASP.NET CORE 中通過依賴註入的方式來使用服務十分的簡單,而在 Console 中,其實也只是稍微繞了個小彎子而已。不管是內置 DI 組件或者第三方的 DI 組件(如Autofac),通過 IServiceCollection 介面我們都可以做到和應用程式的無縫連接。本文將 ...
我們都知道,在 ASP.NET CORE 中通過依賴註入的方式來使用服務十分的簡單,而在 Console 中,其實也只是稍微繞了個小彎子而已。不管是內置 DI 組件或者第三方的 DI 組件(如Autofac),通過 IServiceCollection 介面我們都可以做到和應用程式的無縫連接。本文將在別給出內置組件和第三方組件(主要是Autofac)在 Console 應用程式中的依賴註入實現方式。
1. 在 Console 中使用內置 DI 組件
- 網上已經有幾篇相關的博客講解 Console 中的依賴註入,鏈接都會附於文章末尾。不像 ASP.NET CORE 在應用框架啟動時便將 DI 容器初始化完成並且註入了大部分開發者需要的服務,我們只能從零開始。
// 安裝 DI 組件
Install-Package Microsoft.Extensions.DependencyInjection
// 安裝日誌輸出組件
Install-Package Microsoft.Extensions.Logging.Console
- 添加模擬的應用服務
public interface ICounterAppService
{
void Count(int loops);
}
public class CounterAppService : ICounterAppService
{
private readonly ILogger _logger;
public CounterAppService(ILoggerFactory logger)
{
_logger = logger.CreateLogger<CounterAppService>();
}
public void Count(int loops)
{
for (var i = 0; i < loops; i++)
_logger.LogInformation($"We have got the {i} Loop");
}
}
- Program 中對 DI 組件的初始化和服務的註冊
private static void Main(string[] args)
{
var serviceProvider = new ServiceCollection()
.AddLogging()
.AddSingleton<ICounterAppService, CounterAppService>()
.BuildServiceProvider();
serviceProvider
.GetService<ILoggerFactory>()
.AddConsole(LogLevel.Debug);
var logger = serviceProvider.GetService<ILoggerFactory>()
.CreateLogger<Program>();
logger.LogDebug("Starting application");
var counter = serviceProvider.GetService<ICounterAppService>();
counter.Count(10);
logger.LogDebug("All done!");
}
我們手動創建 serviceProvider 的過程其實就是 ASP.NET CORE 執行 ConfigureServices 方法的過程,同樣的,上述代碼也展示了手動解析 Logger 實例和通過構造函數註入解析 Logger 實例的兩種方式。其中 AddLogging 方法的背後代碼如下所示:
public static IServiceCollection AddLogging(this IServiceCollection services)
{
if (services == null)
throw new ArgumentNullException("services");
services.TryAdd(ServiceDescriptor.Singleton<ILoggerFactory, LoggerFactory>());
services.TryAdd(ServiceDescriptor.Singleton(typeof (ILogger<>), typeof (Logger<>)));
return services;
}
2. 在 Console 中使用第三方 Autofac DI 組件
筆者曾經寫過在 ASP.NET CORE 使用 Autofac 組件的例子,而在 Console 中,註冊流程也沒有什麼變化。以下是 Program 中的代碼
private static void Main(string[] args)
{
var serviceCollection = new ServiceCollection();
serviceCollection.AddLogging();
var containerBuilder = new ContainerBuilder();
// 將原本註冊在內置 DI 組件中的依賴遷移入 Autofac 中
containerBuilder.Populate(serviceCollection);
// 也可以把 ICounterAppService 預先註入到內置 DI 中再使用 Populate 方法遷移
containerBuilder.RegisterType<CounterAppService>().As<ICounterAppService>();
var container = containerBuilder.Build();
var serviceProvider = new AutofacServiceProvider(container);
serviceProvider
.GetService<ILoggerFactory>()
.AddConsole(LogLevel.Debug);
var logger = serviceProvider.GetService<ILoggerFactory>()
.CreateLogger<Program>();
logger.LogDebug("Starting!");
var counter = serviceProvider.GetService<ICounterAppService>();
counter.Count(10);
logger.LogDebug("Done!");
}
同時,Autofac中也提供了諸如 RegisterAssemblyTypes 的方法用於程式集中服務的批量註入,這也是第三方容器的優勢所在。
Using dependency injection in a .Net Core console application
ASP.NET Core Dependency Injection Deep Dive