通常情况下,在使用注入时一个服务接口对应一个实现类,注入方式采用构造函数注入即可,但如果存在多个类实现同一个接口的情况下,则需要根据实际情况来选择不同的实现类。
如以下代码中的MyEmailService和EmailService都实现了IEmailService接口:
public class MyEmailService : IEmailService { public string Send(string Email) { return "My" + Email; } } public class EmailService : IEmailService return Email;
在这种情况,就需要根据不同的情况来选择不同的服务实现了。
Asp.Net Core中自带了容器,同时也可以使用AutoFac替换掉默认的容器,以下为两种方式同名服务的注册实现。
1、使用.net Core自带容器
如果采用自带容器,因为此时我们的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(); }
2、AutoFac中的实现
如果需要在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
到此这篇关于asp.net Core中的同名服务注册的文章就介绍到这了,更多相关asp.net Core服务注册内容请搜索阿兔在线工具以前的文章或继续浏览下面的相关文章希望大家以后多多支持阿兔在线工具!