前言
假设我们有三个Service类实现了同一接口,示例代码如下:
public interface IService { }
public class ServiceA : IService { }
public class ServiceB : IService { }
public class ServiceC : IService { }
我们希望在运行时使用依赖注入指定其具体实现类。
在本文中,我们将演示如何使用HTTP请求设置依赖项设置来实现。
Demo
首先,在Startup.cs中做多个实现类注册:
services.AddHttpContextAccessor();
services.AddTransient<ServiceA>();
services.AddTransient<ServiceB>();
services.AddTransient<ServiceC>();
services.AddTransient<IService>(serviceProvider =>
{
var context = serviceProvider.GetRequiredService<IHttpContextAccessor>();
var containsKey = context.HttpContext?.Request?.Query?.ContainsKey("key");
var key = containsKey.HasValue && containsKey.Value ? context.HttpContext?.Request?.Query?["key"].First(): "A";
switch (key)
{
case "A":
return serviceProvider.GetService<ServiceA>();
case "B":
return serviceProvider.GetService<ServiceB>();
case "C":
return serviceProvider.GetService<ServiceC>();
default:
throw new KeyNotFoundException();
}
});
在这里,我们通过获取HTTP上下文并检查是否定义了key查询字符串参数。
现在,可以向普通方式一样使用IService,示例代码如下:
[ApiController]
[Route("[controller]")]
public class DemoController : ControllerBase
{
private readonly IService _service;
public DemoController(IService service)
{
this._service = service;
}
[HttpGet]
public async Task<string> Get()
{
return _service.GetType().Name;
}
}
通过从查询字符串中读取值,可以控制具体的IService实现类:
结论
当然,我们不仅可以将此策略用于查询字符串中的值,还可以对标头、Body等HTTP请求包含的任何值使用此策略。
想了解更多内容,请关注我的个人公众号”My IO“