热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

.NETCore中的一个接口多种实现的依赖注入与动态选择看这篇就够了

.NETCore中的一个接口多种实现的依赖注入与动态选择看这篇就够了最近有个需求就是一个抽象仓储层接口方法需要SqlServer以及Oracle两种实现方式,为了灵活我在依赖注入的

.NET Core中的一个接口多种实现的依赖注入与动态选择看这篇就够了
最近有个需求就是一个抽象仓储层接口方法需要SqlServer以及Oracle两种实现方式,为了灵活我在依赖注入的时候把这两种实现都给注入进了依赖注入容器中,但是在服务调用的时候总是获取到最后注入的那个方法的实现,这时候就在想能不能实现动态的选择使用哪种实现呢?如果可以的话那么我只需要在配置文件中进行相应的配置即可获取到正确的实现方法的调用,这样的话岂不快哉!今天我们就来一起探讨下实现这种需求的几种实现方式吧。

作者:依乐祝
原文地址:https://www.cnblogs.com/yilezhu/p/10236163.html

代码演示
在开始实现的方式之前,我们先模拟下代码。由于真实系统的结构比较复杂,所以这里我就单独建一个类似的项目结构代码。项目如下图所示:

 

接下来我来详细说下上面的结果作用及代码。

MultiImpDemo.I 这个项目是接口项目,里面有一个简单的接口定义ISayHello,代码如下:

public interface ISayHello
{
string Talk();
}
很简单,就一个模拟讲话的方法。

MultiImpDemo.A 这个类库项目是接口的一种实现方式,里面有一个SayHello类用来实现ISayHello接口,代码如下:

/**
*┌──────────────────────────────────────────────────────────────┐
*│ 描 述:
*│ 作 者:yilezhu
*│ 版 本:1.0
*│ 创建时间:2019/1/7 17:41:33
*└──────────────────────────────────────────────────────────────┘
*┌──────────────────────────────────────────────────────────────┐
*│ 命名空间: MultiImpDemo.A
*│ 类 名: SayHello
*└──────────────────────────────────────────────────────────────┘
*/
using MultiImpDemo.I;
using System;
using System.Collections.Generic;
using System.Text;

namespace MultiImpDemo.A
{
public class SayHello : ISayHello
{
public string Talk()
{
return "Talk from A.SayHello";
}
}
}
MultiImpDemo.B 这个类库项目是接口的另一种实现方式,里面也有一个SayHello类用来实现ISayHello接口,代码如下:

/**
*┌──────────────────────────────────────────────────────────────┐
*│ 描 述:
*│ 作 者:yilezhu
*│ 版 本:1.0
*│ 创建时间:2019/1/7 17:41:45
*└──────────────────────────────────────────────────────────────┘
*┌──────────────────────────────────────────────────────────────┐
*│ 命名空间: MultiImpDemo.B
*│ 类 名: SayHello
*└──────────────────────────────────────────────────────────────┘
*/
using MultiImpDemo.I;
using System;
using System.Collections.Generic;
using System.Text;

namespace MultiImpDemo.B
{
public class SayHello:ISayHello
{
public string Talk()
{
return "Talk from B.SayHello";
}
}
}
MultiImpDemo.Show 这个就是用来显示我们模拟效果的API项目,首选我们在ConfigureServices中加入如下的代码来进行上述两种实现方式的注入:

services.AddTransient();
services.AddTransient();
在api实现里面获取服务并进行模拟调用:

private readonly ISayHello sayHello;

public ValuesController(ISayHello sayHello)
{
this.sayHello = sayHello;
}

// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { sayHello.Talk() };
}
代码很简单对不对?你应该看的懂吧,这时候我们运行起来项目,然后访问API'api/values'这个接口,结果总是显示如下的结果:

 

两种需求对应两种实现
这里有两种业务需求!第一种业务中只需要对其中一种实现方式进行调用,如:业务需要SqlServer数据库的实现就行了。第二种是业务中对这两种实现方式都有用到,如:业务急需要用到Oracle的数据库实现同时也有用到SqlServer的数据库实现,需要同时往这两个数据库中插入相同的数据。下面分别对这两种需求进行解决。

业务中对这两种实现方式都有用到
针对这种情况有如下两种实现方式:

第二种实现方式

其实,在ASP.NET Core中,当你对一个接口注册了多个实现的时候,构造函数是可以注入一个该接口集合的,这个集合里是所有注册过的实现。

下面我们先改造下ConfigureServices,分别注入下这两种实现

services.AddTransient();
services.AddTransient();
接着继续改造下注入的方式,这里我们直接注入IEnumerable如下代码所示:

private readonly ISayHello sayHelloA;
private readonly ISayHello sayHelloB;
public ValuesController(IEnumerable sayHellos)
{
sayHelloA = sayHellos.FirstOrDefault(h => h.GetType().Namespace == "MultiImpDemo.A");
sayHelloB = sayHellos.FirstOrDefault(h => h.GetType().Namespace == "MultiImpDemo.B");
}


// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { sayHelloA.Talk() , sayHelloB.Talk()};
}
然后运行起来看下效果吧

 

利用AddTransient的扩展方法public static IServiceCollection AddTransient(this IServiceCollection services, Func implementationFactory) where TService : class; 然后根据我们的配置的实现来进行服务实现的获取。下面就让我们利用代码来实现一番吧:

services.AddTransient();
services.AddTransient();

services.AddTransient(implementatiOnFactory=>
{
Func accesor = key =>
{
if (key.Equals("MultiImpDemo.A"))
{
return implementationFactory.GetService();
}
else if (key.Equals("MultiImpDemo.B"))
{
return implementationFactory.GetService();
}
else
{
throw new ArgumentException($"Not Support key : {key}");
}
};
return accesor;
});
当然了,既然用到了我们配置文件中的代码,因此我们需要设置下这个配置:

然后我们具体调用的依赖注入的方式需要变化一下:

private readonly ISayHello sayHelloA;
private readonly ISayHello sayHelloB;

private readonly Func _serviceAccessor;

public ValuesController(Func serviceAccessor)
{
this._serviceAccessor = serviceAccessor;

sayHelloA = _serviceAccessor("MultiImpDemoA");
sayHelloB = _serviceAccessor("MultiImpDemoB");
}


// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { sayHelloA.Talk() , sayHelloB.Talk()};
}
然后运行看下效果吧:

 

可以看到A跟B的实现都获取到了!效果实现!

业务只需要对其中一种实现方式的调用
这时候我们可以根据我们预设的配置来动态获取我们所需要的实现。这段话说的我自己都感觉拗口。话不多少,开鲁吧!这里我将介绍三种实现方式。

根据我们的配置文件中设置的key来进行动态的注入。

这种方式实现之前首先得进行相应的配置,如下所示:

"CommonSettings": {
"ImplementAssembly": "MultiImpDemo.A"
}
然后在注入的时候根据配置进行动态的进行注入:

services.AddTransient();
services.AddTransient();
然后在服务调用的时候稍作修改:

private readonly ISayHello sayHello;
public ValuesController(IEnumerable sayHellos,IConfiguration configuration)
{
sayHello = sayHellos.FirstOrDefault(h => h.GetType().Namespace == configuration.GetSection("CommonSettings:ImplementAssembly").Value);
}


// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { sayHello.Talk() };
}
OK,到这里运行一下看下效果吧!然后改下配置文件再看下效果!

 

第二种实现方式,即接口参数的方式这样可以避免上个方法中反射所带来的性能损耗。

这里我们改造下接口,接口中加入一个程序集的属性,如下所示:

public interface ISayHello
{
string ImplementAssemblyName { get; }
string Talk();
}
对应的A跟B中的实现代码也要少做调整:

A:

public string ImplementAssemblyName => "MultiImpDemo.A";

public string Talk()
{
return "Talk from A.SayHello";
}
B:

public string ImplementAssemblyName => "MultiImpDemo.B";

public string Talk()
{
return "Talk from B.SayHello";
}
然后,在实现方法调用的时候稍微修改下:

private readonly ISayHello sayHello;
public ValuesController(IEnumerable sayHellos,IConfiguration configuration)
{
sayHello = sayHellos.FirstOrDefault(h => h.ImplementAssemblyName == configuration.GetSection("CommonSettings:ImplementAssembly").Value);
}


// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { sayHello.Talk() };
}
效果自己运行下看下吧!

第三种实现是根据配置进行动态的注册

首先修改下ConfigureServices方法:

var implementAssembly = Configuration.GetSection("CommonSettings:ImplementAssembly").Value;
if (string.IsNullOrWhiteSpace(implementAssembly)) throw new ArgumentNullException("CommonSettings:ImplementAssembly未配置");
if (implementAssembly.Equals("MultiImpDemo.A"))
{
services.AddTransient();

}
else
{
services.AddTransient();

}
这样的话就会根据我们的配置文件来进行动态的注册,然后我们像往常一样进行服务的调取即可:

private readonly ISayHello _sayHello;
public ValuesController(ISayHello sayHello)
{
_sayHello = sayHello;
}


// GET api/values
[HttpGet]
public ActionResult> Get()
{
return new string[] { _sayHello.Talk() };
}
运行即可得到我们想要的效果!

总结
本文从具体的业务需求入手,根据需求来或动态的进行对应服务的获取,或同时使用两个不同的实现!希望对您有所帮助!如果您有更多的实现方法可以在下方留言,或者加入.NET Core实战千人群跟637326624大伙进行交流,最后感谢您的阅读!
————————————————
版权声明:本文为CSDN博主「依乐祝」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qin_yu_2010/article/details/86150218


推荐阅读
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • Oracle分析函数first_value()和last_value()的用法及原理
    本文介绍了Oracle分析函数first_value()和last_value()的用法和原理,以及在查询销售记录日期和部门中的应用。通过示例和解释,详细说明了first_value()和last_value()的功能和不同之处。同时,对于last_value()的结果出现不一样的情况进行了解释,并提供了理解last_value()默认统计范围的方法。该文对于使用Oracle分析函数的开发人员和数据库管理员具有参考价值。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 本文介绍了Java的集合及其实现类,包括数据结构、抽象类和具体实现类的关系,详细介绍了List接口及其实现类ArrayList的基本操作和特点。文章通过提供相关参考文档和链接,帮助读者更好地理解和使用Java的集合类。 ... [详细]
  • Python SQLAlchemy库的使用方法详解
    本文详细介绍了Python中使用SQLAlchemy库的方法。首先对SQLAlchemy进行了简介,包括其定义、适用的数据库类型等。然后讨论了SQLAlchemy提供的两种主要使用模式,即SQL表达式语言和ORM。针对不同的需求,给出了选择哪种模式的建议。最后,介绍了连接数据库的方法,包括创建SQLAlchemy引擎和执行SQL语句的接口。 ... [详细]
  • Java自带的观察者模式及实现方法详解
    本文介绍了Java自带的观察者模式,包括Observer和Observable对象的定义和使用方法。通过添加观察者和设置内部标志位,当被观察者中的事件发生变化时,通知观察者对象并执行相应的操作。实现观察者模式非常简单,只需继承Observable类和实现Observer接口即可。详情请参考Java官方api文档。 ... [详细]
  • Oracle优化新常态的五大禁止及其性能隐患
    本文介绍了Oracle优化新常态中的五大禁止措施,包括禁止外键、禁止视图、禁止触发器、禁止存储过程和禁止JOB,并分析了这些禁止措施可能带来的性能隐患。文章还讨论了这些禁止措施在C/S架构和B/S架构中的不同应用情况,并提出了解决方案。 ... [详细]
  • 在Oracle11g以前版本中的的DataGuard物理备用数据库,可以以只读的方式打开数据库,但此时MediaRecovery利用日志进行数据同步的过 ... [详细]
  • 本文介绍了Oracle存储过程的基本语法和写法示例,同时还介绍了已命名的系统异常的产生原因。 ... [详细]
  • MySQL中的MVVC多版本并发控制机制的应用及实现
    本文介绍了MySQL中MVCC的应用及实现机制。MVCC是一种提高并发性能的技术,通过对事务内读取的内存进行处理,避免写操作堵塞读操作的并发问题。与其他数据库系统的MVCC实现机制不尽相同,MySQL的MVCC是在undolog中实现的。通过undolog可以找回数据的历史版本,提供给用户读取或在回滚时覆盖数据页上的数据。MySQL的大多数事务型存储引擎都实现了MVCC,但各自的实现机制有所不同。 ... [详细]
author-avatar
娟紫恋蓝_610
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有