07 | 用Autofac增强容器能力:引入面向切面编程(AOP)的能力
这一节讲解使用第三方框架来扩展依赖注入容器
什么情况下需要我们引入第三方容器组件?
大部分情况下,默认的容器组件足够使用
当需要一些非常特殊的场景如下:
1、基于名称的注入:需要把一个服务按照名称来区分它的不同实现的时候
2、属性注入:直接把服务注册到某一个类的属性里面去,而不需要定义构造函数,比如之前的 FromService 和 构造函数入参
3、子容器:可以理解为之前讲过的 Scope,但实际上还可以用第三方的框架实现一些特殊的子容器
4、基于动态代理的 AOP:需要在服务中注入额外的行为的时候,可以用动态代理的能力
.NET Core 的依赖注入框架,它的核心扩展点是 IserviceProviderFactory
第三方的依赖注入容器都是用了这个类来作为扩展点,把自己注入到整个框架里来
也就是说在使用这些依赖注入框架的时候,不需要关注说谁家的特性,谁家的接口是什么样子,只需要关注官方核心的定义就可以了,不需要直接依赖这些框架
源码链接:
https://github.com/witskeeper/geektime/tree/master/samples/DependencyInjectionAutofacDemo
服务
namespace DependencyInjectionAutofacDemo.Services
{public interface IMyService{void ShowCode();}public class MyService : IMyService{public void ShowCode(){Console.WriteLine($"MyService.ShowCode:{GetHashCode()}");}}public class MyServiceV2 : IMyService{/// <summary>/// 用于演示属性注入的方式/// </summary>public MyNameService NameService { get; set; }public void ShowCode(){// 默认情况下,NameService 为空,如果注入成功,则不为空Console.WriteLine($"MyServiceV2.ShowCode:{GetHashCode()},NameService是否为空:{NameService == null}");}}public class MyNameService{}
}
接下来看一下如何集成 Autofac
使用 Autofac 是因为它是 .NET 社区里面最老牌的容器框架之一
它有两个包:
Autofac.Extensions.DependencyInjection
Autofac.Extras.DynamicProxy
引入这两个包,就可以使用它来达到之前说的四种能力
引入这两个包后,需要在 Program 中添加一行代码
.UseServiceProviderFactory(new AutofacServiceProviderFactory())
UseServiceProviderFactory 是用于注册第三方容器的入口
还有一个改动在 Startup 中,我们需要添加一个 ConfigureContainer 方法,它的入参是 Autofac 的 ContainerBuilder
public void ConfigureContainer(ContainerBuilder builder)
现在有两个 ConfigureServices,一个是默认的,一个是 ConfigureContainer
服务注册进默认的容器之后,实际上会被 Autofac 接替,然后执行 ConfigureContainer
Autofac 的注册方式与之前的注册方式不同,先注册具体的实现,然后再告诉它想把它标记为哪个服务的类型,与之前的写法相反
builder.RegisterType<MyService>().As<IMyService>();
接下来是命名注册,当需要把一个服务注册多次,并且用不同命名作为区分的时候,可以用这种方式,入参是一个服务名
builder.RegisterType<MyServiceV2>().Named<IMyService>("service2");
如何使用它呢?
public ILifetimeScope AutofacContainer { get; private set; }public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{// 注册根容器this.AutofacContainer = app.ApplicationServices.GetAutofacRoot();// Autofac 容器获取实例的方式是一组 Resolve 方法var service = this.AutofacContainer.ResolveNamed<IMyService>("service2");service.ShowCode();...
启动程序,输出如下:
MyServiceV2.ShowCode:61566768,NameService是否为空:True