接触 .NET Core 有一段时间了,最大的感受无外乎无所不在的依赖注入,以及抽象化程度更高的全新框架设计。想起三年前 Peter 大神手写 IoC 容器时的惊艳,此时此刻,也许会有不一样的体会。的确,那个基于字典实现的 IoC 容器相当“简陋”,就像 .NET Core 里的依赖注入,默认(原生)都是采用构造函数注入的方式,可其实从整个依赖注入的理论上而言,属性注入和方法注入的方式,同样是依赖注入的实现方式啊。最近一位朋友找我讨论,.NET Core 里该如何实现 Autowried,这位朋友本身是 Java 出身,一番攀谈了解到原来是指属性注入啊。所以,我打算用两篇博客来聊聊 .NET Core 中的原生 DI 的扩展,而今天这篇,则单讲基于名称的注入的实现。
Autofac是一个非常不错的 IoC 容器,通常我们会使用它来替换微软内置的 IoC 容器。为什么要这样做呢?其实,微软在其官方文档中早已给出了说明,即微软内置的 IoC 容器实际上是不支持以下特性的: 属性注入、基于名称的注入、子容器、自定义生存期管理、对迟缓初始化的 Func支持、基于约定的注册。这是我们为什么要替换微软内置的 IoC 容器的原因,除了 Autofac 以外,我们还可以考虑 Unity 、Castle 等容器,对我个人而言,其实最需要的一个功能是“扫描”,即它可以针对程序集中的组件或者服务进行自动注册。这个功能可以让人写起代码更省心一点,果然,人类的本质就是让自己变得更加懒惰呢。好了,话题拉回到本文主题,我们为什么需要基于名称的注入呢?它其实针对的是“同一个接口对应多种不同的实现”这种场景。
var services = new ServiceCollection(); services.AddTransient<ISayHello, ChineseSayHello>(); services.AddTransient<ISayHello, EnglishSayHello>(); var serviceProvider = services.BuildServiceProvider(); var sayHello = serviceProvider.GetRequiredService<ISayHello>();
var sayHelloFactory = _serviceProvider.GetRequiredService<Func<string, ISayHello>>(); var chineseSayHello = sayHelloFactory("Chinese"); var englishSayHello = sayHelloFactory("English");
public TService GetService<TService>(string serviceName) { if(!_registrations.TryGetValue(serviceName, outvar implementationType)) thrownew ArgumentException($"Service \"{serviceName}\" is not registered in container"); return (TService)_serviceProvider.GetService(implementationType); } }
可以注意到,我们这里用一个字典来维护名称和类型间的关系,一切仿佛又回到三年前 Peter 大神手写 IoC 的那个下午。接下来,我们定义一个INamedServiceProviderBuilder, 它可以让我们使用链式语法注册服务:
publicvoidBuild() { _services.AddTransient<INamedServiceProvider>(sp => new NamedServiceProvider(sp, _registrations)); }
public INamedServiceProviderBuilder AddNamedService<TImplementation>(string serviceName, ServiceLifetime lifetime) where TImplementation : class { switch (lifetime) { case ServiceLifetime.Transient: _services.AddTransient<TImplementation>(); break; case ServiceLifetime.Scoped: _services.AddScoped<TImplementation>(); break; case ServiceLifetime.Singleton: _services.AddSingleton<TImplementation>(); break; }
public INamedServiceProviderBuilder TryAddNamedService<TImplementation>(string serviceName, ServiceLifetime lifetime) where TImplementation : class { switch (lifetime) { case ServiceLifetime.Transient: _services.TryAddTransient<TImplementation>(); break; case ServiceLifetime.Scoped: _services.TryAddScoped<TImplementation>(); break; case ServiceLifetime.Singleton: _services.TryAddSingleton<TImplementation>(); break; }