首页 > 其他分享 >.NET Core依赖注入与Autofac注入介绍

.NET Core依赖注入与Autofac注入介绍

时间:2023-07-27 12:34:01浏览次数:46  
标签:Core Autofac pay builder public Pay IPay 注入

0 前言

本文主要介绍了ASP.NET Core自带的依赖注入框架的用法,然后针对原生框架的不足,介绍了更加完备的autofac框架的集成和使用。

1 .NET Core原生DI框架

.Net Core自带一个依赖注入的框架,使用起来很是方便,不多说,先从简单示例做起。

1.1 简单示例

以ASP.NET Core web的API项目为例:

首先,定义接口IPay和实现类WxPay:

public interface IPay
{
    void Pay();
}

public void Pay()
{
    Console.WriteLine("wxpay");
}

然后在Startup类中的ConfigureService方法中进行注册:

 services.AddScoped<IPay, WxPay>();

接着,就可以在Contoller中使用构造函数注入:

[ApiController]
[Route("[controller]")]
public class WeatherForecastController : ControllerBase
{
    private readonly IPay pay;

    public WeatherForecastController(IPay pay)
    {
        this.pay = pay;
    }

    [HttpGet]
    public string Get()
    {
        return pay.Pay();
    }
}

最后,启动程序,看到输出:

wxpay

1.2 生存周期

* Singleton: 单例模式,即在整个应用程序生存期内只会有一个实例
* Scoped:作用域对象在一个客户端请求中是相同的,但在多个客户端请求中是不同的
* Transient:暂时性对象始终不同,无论是不是同一个请求(同一个请求里的不同服务)同一个客户端,每次都是创建新的实例

对Transient的模式,我个人还没怎么理解,暂时先放在这里。

按我个人的想法,正常情况下使用Scoped模式,保证在当前请求内保证一致即可。而Transient模式的场景感觉就是在同一个请求中,获取服务后进行业务操作时,获取相关服务的数据状态发生了变更,因此在再次使用时,需要重新获取服务状态,因此使用暂时模式。

这个可以参见一下几个理解:

上传失败

另外,看官方文档上的示例:MSDN

1.3 同一接口多个实现

对于IPay接口来说,除了WxPay我们还有AliPay的支付方式,

public string Pay()
{
    return "AliPay";
}

此时在DI注入时就需要将两个都注入:

services.AddScoped<IPay, WxPay>();
services.AddScoped<IPay, AliPay>();

然后我们再运行代码,会发现页面此时的输入变成了:AliPay。 说明这种情况下注入的是最后一次的接口实现。

那么如何获取到其他的实现呢?

我们可以通过注入IEnumerable<IPay>来实现。

[ApiController]
[Route("[controller]")]
public class WeatherForecastController : ControllerBase
{
    private readonly IPay pay;
    private readonly IEnumerable<IPay> payList;

    public WeatherForecastController(IPay pay, IEnumerable<IPay> payList)
    {
        this.pay = pay;
        this.payList = payList;
    }

    [HttpGet]
    public string Get()
    {
        var sb = new StringBuilder();
        foreach (var pay in payList)
        {
            sb.AppendLine(pay.Pay());
        }

        return sb.ToString();
    }
}

此时,运行后得到的输出是:

wxpay
AliPay

这样根据注入payList的类型就能得到所对应的服务实现。当然也可以使用一个工厂类将所有实现注入,然后通过服务类型的方式进行获取。

不过,更推荐使用下面要介绍的Autofac来命名实现。

2 Autofac

Autofac是.net语言的老牌依赖注入框架了,详细的见其官网,我们只管拿来用就完事了。

2.1 简单配置

  • 安装autofac

这个不多说了,在Nuget Packge管理界面查找安装即可。

  • Program.cs启动配置

需要在CreateHostBuilder中对autofac进行配置,使用UseServiceProviderFactory(new AutofacServiceProviderFactory()), 将服务提供程序注入到.net core中:

public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.UseServiceProviderFactory(new AutofacServiceProviderFactory())
    .ConfigureWebHostDefaults(webBuilder =>
    {
        webBuilder.UseStartup<Startup>();
    });
  • 在Startup类中进行接口注入

为了代码结构的阅读方便,我们使用一个单独的注入类DependencyRegister,在类中实现对接口的注入:

public static class DependencyRegister
{
    public static ContainerBuilder Register(ContainerBuilder builder)
    {
        builder.RegisterType<MyService>().As<IMyInterface>();
        return builder;
    }
}

注入容器我们需要添加ConfigureContainer方法,然后在Startup.cs中添加:

public void ConfigureContainer(ContainerBuilder builder)
{
    DependencyRegister.Register(builder);
}

至此,MyService作为IMyInterface的实现就在容器中,在controller类中通过像.net core中一样的构造注入方式就能得到对应的实例。

2.2 命名:同一接口多个实现

还是以上面的IPay为例,我们在DependencyRegister的Register()中添加:

builder.RegisterType<WxPay>().Named<IPay>(typeof(WxPay).Name);
builder.RegisterType<AliPay>().Named<IPay>(typeof(AliPay).Name);

上面表示往容器中分别注册了两个名为"WxPay","AliPay"的IPay接口实现。

autofac可以通过注入IComponentContext来获取上下文,然后从上下文中通过名称来获取对应的接口实现。

如我们新增一个homeController,代码如下:

[ApiController]
[Route("home")]
public class HomeController : ControllerBase
{
    private readonly IPay wxPay;
    private readonly IPay aliPay;

    public HomeController(IComponentContext componentContext)
    {

        builder.RegisterType<WxPay>().Named<IPay>(typeof(WxPay).Name);
        builder.RegisterType<AliPay>().Named<IPay>(typeof(AliPay).Name);
    }

    [Route("test")]
    [HttpGet]
    public string Test()
    {
        var str = new StringBuilder();
        str.AppendLine(wxPay.Pay());
        str.AppendLine(aliPay.Pay());
        return str.ToString();
    }
}

运行代码,访问/home/test,同样能见到两个接口的输出。

2.3 按条件批量注入

日常开发中,新增接口和实现是很多的,如果每次新增都需要手动添加的话是比较繁琐的。于是我们可以用通过通配符的方式来实现自动批量注入,如下面查询以Pay为结尾的类:

var assembly1 = Assembly.GetExecutingAssembly();
builder.RegisterAssemblyTypes(assembly1)
    .PublicOnly()
    .Where(r => r.Name.EndsWith("Pay"))
    .AsImplementedInterfaces();

标签:Core,Autofac,pay,builder,public,Pay,IPay,注入
From: https://www.cnblogs.com/Alex80/p/17584641.html

相关文章

  • 5、其他位置的注入
    limit注入 limit注入,这个词是受注入位置划分的。whereid=xx 后面的,也可以叫做where注入。 limitunion注入没有orderby的情况SQL:select*fromuserswhereid<=100limit1unionselect1,2,3--,1?p=1unionselect1,2,3--+?p=1unionselect1,(S......
  • asp.net core之Host
    Host简介在ASP.NETCore中,Host是一个托管应用程序的宿主环境。它提供了一种统一的方式来启动和运行应用程序,无论是在开发环境中还是在生产环境中。Host负责处理应用程序的生命周期、配置和依赖项管理等任务,使开发人员能够专注于应用程序的业务逻辑。Host是通过使用IHostBuilder......
  • 网络安全之SQL注入基于DVWA平台
    弱口令SQL注入万能密码admin'--'admin'#万能用户名xxx'or1=1limit1---脱库一库:information_schema三表:schemata表:存放所有数据库信息tables表:存放所有表信息columns表:存放所有字段信息六字段:schemata表的schema_name字段:存放具体的数据库名......
  • ASP.NET Core Identity 系列之一
    ASP.NETCoreIdentity提供给我们一组工具包和API,它能帮助我们应用程序创建授权和认证功能,也可以用它创建账户并使用用户名和密码进行登录,同时也提供了角色和角色管理功能。ASP.NETCoreIdentity使用SQLServer/第三方数据库存储用户名和密码,角色和配置数据这系列中我们主要使用......
  • ASP.NET Core Identity 系列之一
    ASP.NETCoreIdentity提供给我们一组工具包和API,它能帮助我们应用程序创建授权和认证功能,也可以用它创建账户并使用用户名和密码进行登录,同时也提供了角色和角色管理功能。ASP.NETCoreIdentity使用SQLServer/第三方数据库存储用户名和密码,角色和配置数据这系列中我们主要使用VS......
  • >.NET Core|--Quartz.Net|--控制台应用程序
    前言#VisualStudio版本MicrosoftVisualStudioProfessional2022(64位)#Quartz.Net版本Quartz.Net3.6.3#C#语言版本(别小看这个,否则我下面的代码在你那里很可能跑不起来)10.0安装Quartz.Net我是直接通过Nuget包管理器安装的其它安装方式,也可以使用命......
  • quartz.net core
    如何使用Quartz.NetCore介绍Quartz.NetCore是一个开源的任务调度库,可以帮助开发者实现定时任务的调度和执行。本文将指导你如何使用Quartz.NetCore。首先,让我们看一下整个实现过程的流程图。实现步骤下面是使用Quartz.NetCore实现定时任务的步骤:步骤描述步......
  • asp.net core之依赖注入
    依赖注入概念ASP.NETCore支持依赖关系注入(DI)软件设计模式,这是一种在类及其依赖关系之间实现控制反转(IoC)的技术。按照官方文档的描述:依赖关系注入通过以下方式解决了这些问题:使用接口或基类将依赖关系实现抽象化。在服务容器中注册依赖关系。ASP.NETCore提供了一......
  • NetCore 控制台 上下文 注入 dbcontext
    publicclassProgram{privatestaticTestDBContext_tContext;privatestaticvoidMain(string[]args){Startup();vardd=_tContext.User.Take(1).ToList();//e();}privatestaticvoidStartup()......
  • 特征选择 - Fisher Score
    特征选择的目的在理想情况下,特征选择想要达到以下效果:简化模型以提高可解释性:通过减少特征的数量,模型变得更简单,更容易理解。这对于那些需要理解模型如何做出预测的领域(如医疗或信贷评分)非常重要。改进模型性能:通过消除无关或冗余的特征,模型的预测性能可能会得到提高。这是......