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

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

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






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





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



作者:依乐祝

原文地址:https://www.cnblogs.com/yilezhu/p/10236163.html



代码演示


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


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


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




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


        public interface ISayHello
    {
    string Talk();
    }

    很简单,就一个模拟讲话的方法。




  2. 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";
    }
    }
    }



  3. 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";
    }
    }
    }



  4. MultiImpDemo.Show 这个就是用来显示我们模拟效果的API项目,首选我们在ConfigureServices中加入如下的代码来进行上述两种实现方式的注入:


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



  5. 在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'这个接口,结果总是显示如下的结果:


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




两种需求对应两种实现


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


业务中对这两种实现方式都有用到


针对这种情况有如下两种实现方式:




  1. 第二种实现方式


    其实,在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()};
    }

    然后运行起来看下效果吧


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




  2. 利用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()};
    }

    然后运行看下效果吧:


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


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




业务只需要对其中一种实现方式的调用


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




  1. 根据我们的配置文件中设置的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,到这里运行一下看下效果吧!然后改下配置文件再看下效果!


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




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


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


    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() };
    }

    效果自己运行下看下吧!




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


    首先修改下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大伙进行交流,最后感谢您的阅读!














posted @ 2019-01-07 22:50 依乐祝 阅读(...) 评论(...) 编辑 收藏






推荐阅读
  • Startup 类配置服务和应用的请求管道。Startup类ASP.NETCore应用使用 Startup 类,按照约定命名为 Startup。 Startup 类:可选择性地包括 ... [详细]
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • 深入理解 .NET 中的中间件
    中间件是插入到应用程序请求处理管道中的组件,用于处理传入的HTTP请求和响应。它在ASP.NET Core中扮演着至关重要的角色,能够灵活地扩展和自定义应用程序的行为。 ... [详细]
  • 本文介绍如何在Spring Boot项目中集成Redis,并通过具体案例展示其配置和使用方法。包括添加依赖、配置连接信息、自定义序列化方式以及实现仓储接口。 ... [详细]
  • 在 Android 开发中,通过 Intent 启动 Activity 或 Service 时,可以使用 putExtra 方法传递数据。接收方可以通过 getIntent().getExtras() 获取这些数据。本文将介绍如何使用 RoboGuice 框架简化这一过程,特别是 @InjectExtra 注解的使用。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • 在维护公司项目时,发现按下手机的某个物理按键后会激活相应的服务,并在屏幕上模拟点击特定坐标点。本文详细介绍了如何使用ADB Shell Input命令来模拟各种输入事件,包括滑动、按键和点击等。 ... [详细]
  • MySQL 数据库迁移指南:从本地到远程及磁盘间迁移
    本文详细介绍了如何在不同场景下进行 MySQL 数据库的迁移,包括从一个硬盘迁移到另一个硬盘、从一台计算机迁移到另一台计算机,以及解决迁移过程中可能遇到的问题。 ... [详细]
  • MySQL索引详解与优化
    本文深入探讨了MySQL中的索引机制,包括索引的基本概念、优势与劣势、分类及其实现原理,并详细介绍了索引的使用场景和优化技巧。通过具体示例,帮助读者更好地理解和应用索引以提升数据库性能。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 实体映射最强工具类:MapStruct真香 ... [详细]
  • 本文探讨了在 ASP.NET MVC 5 中实现松耦合组件的方法。通过分离关注点,应用程序的各个组件可以更加独立且易于维护和测试。文中详细介绍了依赖项注入(DI)及其在实现松耦合中的作用。 ... [详细]
  • 本文将深入探讨如何在不依赖第三方库的情况下,使用 React 处理表单输入和验证。我们将介绍一种高效且灵活的方法,涵盖表单提交、输入验证及错误处理等关键功能。 ... [详细]
author-avatar
自由常飞_337
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有