咨询区
LP13
我有三个 Service
类实现了同一个接口,参考代码如下:
public interface IService { }
public class ServiceA : IService { }
public class ServiceB : IService { }
public class ServiceC : IService { }
我知道像其他的 DI 容器,比如 Unity 是在注册实现类的时候通过不同的key来区分,在 ASP.NET Core
中做 AddService 时并没有看到 key
或 name
参数的重载方法。
public void ConfigureServices(IServiceCollection services){ // How do I register services of the same interface? }public MyController:Controller{public void DoSomething(string key){ // How do I resolve the service by key?}}
请问我该如何实现呢?
回答区
Miguel A. Arilla
我也遇到了这种场景,不过我是用 Func
做了一个折中方案。
首先:定义一个 委托
。
public delegate IService ServiceResolver(string key);
然后:在 Startup.cs
中做多个实现类注册。
services.AddTransient<ServiceA>();
services.AddTransient<ServiceB>();
services.AddTransient<ServiceC>();services.AddTransient<ServiceResolver>(serviceProvider => key =>
{switch (key){case "A":return serviceProvider.GetService<ServiceA>();case "B":return serviceProvider.GetService<ServiceB>();case "C":return serviceProvider.GetService<ServiceC>();default:throw new KeyNotFoundException(); // or maybe return null, up to you}
});
后期:如果想使用的话,通过 key
从 Func 中提取具体的实例,参考代码如下:
public class Consumer
{private readonly IService _aService;public Consumer(ServiceResolver serviceAccessor){_aService = serviceAccessor("A");}public void UseServiceA(){_aService.DoTheThing();}
}
为了做演示目的,我这里只用了 string 作为解析,你可以根据自己的场景具体实现。
点评区
在 Asp.Net Core
中是不支持这种多注册的,这就有点尴尬了,Miguel A. Arilla
大佬提供的方案不错,学习了。