通常情况下,在使用注入时一个服务接口对应一个实现类,注入方式采用构造函数注入即可,但如果存在多个类实现同一个接口的情况下,则需要根据实际情况来选择不同的实现类。
如以下代码中的MyEmailService和EmailService都实现了IEmailService接口:
public class MyEmailService : IEmailService { public string Send(string Email) { return "My" + Email; } } public class EmailService : IEmailService { public string Send(string Email) { return Email; } }
在这种情况,就需要根据不同的情况来选择不同的服务实现了。
Asp.Net Core中自带了容器,同时也可以使用AutoFac替换掉默认的容器,以下为两种方式同名服务的注册实现。
如果采用自带容器,因为此时我们的IEmailService有多个实现类,因此需要注入IServiceProvider,然后通过IServiceProvider的GetServices复数版本来获取多个服务,此时可以顺序调用多个实现服务,如果需要区别其中的某一个服务,可以通过type加以区分。
[Route("api/[controller]")] [ApiController] public class ValuesController : ControllerBase { private readonly IEnumerable<IEmailService> emailService; private readonly ILogger<ValuesController> logger; /// <summary> /// 注入IServiceProvider获取服务 /// </summary> /// <param name="serviceProvider"></param> /// <param name="logger"></param> public ValuesController(IServiceProvider serviceProvider, ILogger<ValuesController> logger) { var service = serviceProvider.GetServices<IEmailService>(); //获取服务 this.emailService = service; this.logger = logger; } [HttpGet] public IActionResult Send(string email) { //可以遍历服务 foreach (var emailService in emailService) { if(emailService.GetType() == typeof(First_EmailService)) { //控制台输出调用日志 logger.LogInformation(emailService.Send(email)); } } return Ok(); } }
如果需要在Autofac中解析不同的服务,需要用到Named实现。首先,需要获取Autofac的容器,即IContainer实例。获取的位置有两种:
(1)在ConfigureContainer中的注册模块类中进行获取,相关代码如下:
public class ConfigureAutofac : Autofac.Module { private static IContainer _container; protected override void Load(ContainerBuilder containerBuilder) { //注册两个不同命名的服务 containerBuilder.RegisterType<EmailService>().Named<IEmailService>("one"); containerBuilder.RegisterType<MyEmailService>().Named<IEmailService>("two"); //需要在回调用获取容器 containerBuilder.RegisterBuildCallback(container => { _container = (IContainer)container; var one = _container.ResolveNamed<IEmailService>("two"); one.Send("one"); }); } }
(2)上述服务是Autofac的模块配置中实现,也可以在Startup中实现,需要调用Autofac的GetAutofacRoot方法
/// <summary> /// ConfigureServices将默认的容器注入进去,Autofac会接替默认的,然后执行ConfigureContainer /// </summary> /// <param name="builder"></param> public void ConfigureContainer(ContainerBuilder builder) { builder.RegisterType<MyService>().As<IMyService>(); } //注册根容器 public ILifetimeScope AutofacContainer { get; private set; } public void Configure(IApplicationBuilder app, IWebHostEnvironment env) { this.AutofacContainer = app.ApplicationServices.GetAutofacRoot(); var serviceName = this.AutofacContainer.Resolve<IMyService>(); serviceName.ShowCode(); //... }
(3)如果需要在控制器中实现不同名称的服务调用,则需要在控制器中注入IApplicationBuilder接口,但直接注入会报未解析服务的错误,需要手动在ConfigureServices中配置一下
services.AddSingleton<IApplicationBuilder, ApplicationBuilder>();
之后就可以在控制器中正常注入了
[Route("api/[controller]")] [ApiController] public class ValuesController : ControllerBase { private readonly IApplicationBuilder app; private readonly ILogger<ValuesController> logger; public ILifetimeScope AutofacContainer { get; private set; } public ValuesController(IApplicationBuilder app, ILogger<ValuesController> logger) { this.app = app; this.logger = logger; } [HttpGet] public IActionResult Send(string email) { this.AutofacContainer = app.ApplicationServices.GetAutofacRoot(); var serviceName = this.AutofacContainer.ResolveNamed<IEmailService>("one"); logger.LogInformation(serviceName.Send(string.Empty)); return Ok(); } }
参考:https://blog.csdn.net/hahahzzzzz/article/details/118684998