在.NET Core中处理一个接口多个不同实现的依赖注入问题
前言
近段时间在准备公司的技术分享,所以这段时间将大部分时间放在准备分享内容上去了。博客也就停了一下下。
在.NET Core中处理依赖注入问题时,往往是定义好了一个操作规范的接口,会有N多个基于不同技术的实现,根据实际情况在项目中去使用某一个实现。
但是偶尔会出现这样的情况,在某一个地方,需要同时使用到两种或两种以上的实现,这个时候我们要怎么处理呢?
借助Autofac等第三方组件时,是可以很容易的实现,但是在写一些基础类库时会避免直接引用太多依赖组件。
所以这里是只用微软自带的DI(Microsoft.Extensions.DependencyInjection)去处理。
例子引入
现在有一个接口和两个实现类。
public interface IDemoService
{
string Get();
}
public class DemoServiceA : IDemoService
{
public string Get()
{
return "Service A";
}
}
public class DemoServiceB : IDemoService
{
public string Get()
{
return "Service B";
}
}
常规的方法,我们先在Startup中的ConfigureServices方法中添加我们的service。
public void ConfigureServices(IServiceCollection services)
{
services.AddSingleton<IDemoService, DemoServiceA>();
services.AddSingleton<IDemoService, DemoServiceB>();
services.AddMvc();
}
然后在控制器中使用
private IDemoService _serviceA;
private IDemoService _serviceB;
public ValuesController(IDemoService serviceA, IDemoService serviceB)
{
_serviceA = serviceA;
_serviceB = serviceB;
}
// GET api/values
[HttpGet]
public string Get()
{
return $"{_serviceA.Get()}-{_serviceB.Get()}";
}
我们的预期结果是:Service A-Service B
,可是上面代码的实际结果却并不像我们想的那么简单!!
可以看到这里输出的都是Service B
,连Service A
的影子都没有看到。
其实,从代码都可以看出来,它只能拿到其中一个Service的实现类!
那么我们要息怎样处理才能达到我们想要的效果呢?
其实思路比较简单,上面导致不能拿到对应实现类,本质上来讲应该说是它区分不了那个才是想要的!我们想个办法让它能区分就好了。
处理方法
给我们的Service起个别名!
先是Startup中的ConfigureServices方法。
public void ConfigureServices(IServiceCollection services)
{
services.AddSingleton<DemoServiceA>();
services.AddSingleton<DemoServiceB>();
services.AddSingleton(factory =>
{
Func<string, IDemoService> accesor = key =>
{
if (key.Equals("ServiceA"))
{
return factory.GetService<DemoServiceA>();
}
else if (key.Equals("ServiceB"))
{
return factory.GetService<DemoServiceB>();
}
else
{
throw new ArgumentException($"Not Support key : {key}");
}
};
return accesor;
});
services.AddMvc();
}
这里并没有直接向上面那样一次性指定接口和对应的实现类,而是用了AddSingleton的另一个重载方法。
- 先将实现类注册一下
- 然后再注册一下
Func<string, IDemoService>
先来说说这个Func<string, IDemoService>
里面的string和IDemoService都分别代表什么。
- string 毫无疑问就是我们上面说到的别名
- IDemoService 这个就是我们要用的Service
核心在于,factory参数是IServiceProvider类型的!所以我们可以根据这个factory去找到我们前面注册的实现类。这样解释一下,是不是就清晰了呢?
然后再来看看在控制器上面怎么用。
private IDemoService _serviceA;
private IDemoService _serviceB;
private readonly Func<string, IDemoService> _serviceAccessor;
public ValuesController(Func<string, IDemoService> serviceAccessor)
{
this._serviceAccessor = serviceAccessor;
_serviceA = _serviceAccessor("ServiceA");
_serviceB = _serviceAccessor("ServiceB");
}
// GET api/values
[HttpGet]
public string Get()
{
return $"{_serviceA.Get()}-{_serviceB.Get()}";
}
最后看看结果是不是和我们的预期一样。
结果与预期一致。
总结
一对一,或许是最好的方法,也是最为理想的,这样能避开很多不必要的问题。但是现实中总会出现特殊情况,面对这些特殊情况,我们也是需要能够重容的面对。
如果您有更好的处理方法,也可以留言讨论。
文中的示例代码 DIDemo
在.NET Core中处理一个接口多个不同实现的依赖注入问题的更多相关文章
- ExpandoObject与DynamicObject的使用 RabbitMQ与.net core(一)安装 RabbitMQ与.net core(二)Producer与Exchange ASP.NET Core 2.1 : 十五.图解路由(2.1 or earler) .NET Core中的一个接口多种实现的依赖注入与动态选择看这篇就够了
ExpandoObject与DynamicObject的使用 using ImpromptuInterface; using System; using System.Dynamic; names ...
- .NET Core中的一个接口多种实现的依赖注入与动态选择看这篇就够了
最近有个需求就是一个抽象仓储层接口方法需要SqlServer以及Oracle两种实现方式,为了灵活我在依赖注入的时候把这两种实现都给注入进了依赖注入容器中,但是在服务调用的时候总是获取到最后注入的那个 ...
- ASP.NET Core Web 应用程序系列(二)- 在ASP.NET Core中使用Autofac替换自带DI进行批量依赖注入(MVC当中应用)
在上一章中主要和大家分享在MVC当中如何使用ASP.NET Core内置的DI进行批量依赖注入,本章将继续和大家分享在ASP.NET Core中如何使用Autofac替换自带DI进行批量依赖注入. P ...
- 如何在ASP.NET Core中实现一个基础的身份认证
注:本文提到的代码示例下载地址> How to achieve a basic authorization in ASP.NET Core 如何在ASP.NET Core中实现一个基础的身份认证 ...
- 在ASP.NET Core中实现一个Token base的身份认证
注:本文提到的代码示例下载地址> How to achieve a bearer token authentication and authorization in ASP.NET Core 在 ...
- [转]如何在ASP.NET Core中实现一个基础的身份认证
本文转自:http://www.cnblogs.com/onecodeonescript/p/6015512.html 注:本文提到的代码示例下载地址> How to achieve a bas ...
- [转]NET Core中实现一个Token base的身份认证
本文转自:http://www.cnblogs.com/Leo_wl/p/6077203.html 注:本文提到的代码示例下载地址> How to achieve a bearer token ...
- NET Core中实现一个Token base的身份认证
NET Core中实现一个Token base的身份认证 注:本文提到的代码示例下载地址> How to achieve a bearer token authentication and au ...
- Dora.Interception,为.NET Core度身打造的AOP框架 [4]:与依赖注入框架的无缝集成
Dora.Interception最初的定位就是专门针对.NET Core的AOP框架,所以在整个迭代过程中我大部分是在做减法.对于.NET Core程序开发来说,依赖注入已经成为无处不在并且“深入骨 ...
随机推荐
- WPF字典集合类ObservableDictionary
WPF最核心的技术优势之一就是数据绑定.数据绑定,可以通过对数据的操作来更新界面. 数据绑定最经常用到的是ObservableCollection<T> 和 Dictionary<T ...
- 浅讲JUnit
JUnit单元简介: JUnit ----是一个开发源码的java测试框架,用于编写和运行可重复的测试,它是用于单元测试框架体系xUnit的一个实例, 用于java语言,主要用于白盒测试,回 ...
- 02-Nginx+MySQL+PHP7
[安装Nginx] #先安装如下包 yum install gcc gcc-c++ kernel-devel yum -y install pcre-devel openssl openssl-dev ...
- MTU介绍以及在windows和linux下怎么设置MTU值
最大传输单元MTU(Maximum Transmission Unit,MTU)是指一种通信协议的某一层上面所能通过的最大数据包大小(以字节为单位).最大传输单元这个参数通常与通信接口有关(网络接口卡 ...
- Python自动化--语言基础2--运算符、格式化输出、条件语句、循环语句、列表、元组
运算符包括:算术运算符.比较运算符.赋值运算符.逻辑运算符.成员运算符.身份运算符 算术运算符 % 取模(余数) // 取相除的整数部分 / (5/2=2.5) 比较运算符 == 等于 ! ...
- js的继承实现
1.原型链继承 1.创建父类对象2.创建子类函数对象3.将父类的实例对象赋值给子类的原型4.将子类的原型属性的构造函数设置为 子类本身 function Person(name) { this.nam ...
- 相对路径和绝对路径的问题"/"带不带斜杠
带有"/"是绝对路径,为项目的上下文目录是从工程开始的路径 不带的话是一个相对路径,相对于执行该代码的目录文件结构
- UVA1619 栈维护递增序列
先说这题的关键性质:每一个数应该只会计算一次,它有一个最小区间[L,R],即它在这个区间内是最小的,最小区间内任何包含它的子区间都不会大于F(L,R)=(a[L]+...+a[R])*min(a[l] ...
- 设置TCP_USER_TIMEOUT参数来判断tcp连接是否断开
[TOC] 1. bug描述 前段时间遇到这样的一个问题,openstack一个控制节点宕机后,在宕机后一段时间内创建的虚拟机,一直卡在创建中的状态.有的甚至要等到16分钟之后虚拟机才会切换到下一个状 ...
- Five nines
Five nines, commonly taken to mean "99.999%", may refer to: 高可用 High availability of serv ...