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

本文不介绍IoC和DI的概念,如果你对Ioc之前没有了解的话,建议先去搜索一下相关的资料
(.net core 3.0已经不需要这种做法了)

这篇文章将简单介绍一下AutoFac的基本使用以及在asp .net core中的应用

Autofac介绍
组件的三种注册方式
反射
现成的实例(new)
lambda表达式 (一个执行实例化对象的匿名方法)
下面是一些简短的示例,我尽可能多的列出来一些常用的注册方式,同时在注释中解释下“组件”、“服务”等一些名词的含义

// 创建注册组件的builder
var builder = new ContainerBuilder();

//根据类型注册组件 ConsoleLogger 暴漏服务:ILogger
builder.RegisterType().As();

//根据类型注册组件 ConsoleLogger,暴漏其实现的所有服务(接口)
builder.RegisterType().AsImplementedInterfaces();

// 根据实例注册组件 output 暴漏服务:TextWriter
var output = new StringWriter();
builder.RegisterInstance(output).As();

//表达式注册组件,这里我们是在构造函数时传参->“musection” 暴漏服务:IConfigReader
builder.Register(c =new ConfigReader(“mysection”)).As();

//表达式注册组件,解析时传参
var service = scope.Resolve(
new NamedParameter(“section”, “mysection”));

//反射注册组件,直接注册了ConsoleLogger类(必须是具体的类),如果ConsoleLogger有多个构造函数,将会取参数最多的那个构造函数进行实例化
builder.RegisterType();

//反射注册组件,手动指定构造函数,这里指定了调用 MyComponent(ILogger log,IConfigReader config)的构造函数进行注册
builder.RegisterType()
.UsingConstructor(typeof(ILogger), typeof(IConfigReader));

//注册MySingleton类中的静态变量"Instance",ExternallyOwned()函数指定自己控制实例的生命周期,而不是由autofac自动释放
builder.RegisterInstance(MySingleton.Instance).ExternallyOwned();

//一个组件暴漏两个服务
builder.RegisterType().As().As();

//注册当前程序集中以“Service”结尾的类
builder.RegisterAssemblyTypes(System.Reflection.Assembly.GetExecutingAssembly()).Where(t => t.Name.EndsWith(“Service”)).AsImplementedInterfaces();
//注册"MyApp.Repository"程序集中所有的类
builder.RegisterAssemblyTypes(GetAssembly(“MyApp.Repository”)).AsImplementedInterfaces();

//构建一个容器完成注册
var rootcontainer = builder.Build();

//可以通过下面这种方式手动获取IConfigReader 的实现类
//这种手动解析的方式需要 从生命周期作用域内获取组件,以保证组件最终被释放
//不要直接从根容器rootcontainer中解析组件,很有可能会导致内存泄漏
using(var scope = rootcontainer.BeginLifetimeScope())
{
var reader = scope.Resolve();
}
如果不止一个组件暴露了相同的服务, Autofac将使用最后注册的组件作为服务的提供方。 想要覆盖这种行为, 在注册代码后使用 PreserveExistingDefaults() 方法修改

生命周期
using(var scope = rootcontainer.BeginLifetimeScope())

上面的这段代码创建了一个生命周期作用域

生命周期作用域是可释放的,在作用域内解析的组件一定要保证在using之内使用或者最后手动调用组件的Dispose()函数

避免引用类的生命周期大于被引用类的生命周期 :如service 引用 repository 如果service的生命周期为单例,repository的生命周期为perrequest。service不会释放,所以最终会造成相关的repository始终无法释放的情况(Captive Dependencies)

虽然我们需要尽可能的避免直接从根容器解析组件,但总有例外的情况,对于非单例的组件,一定不要忘记调用组件的Dispose函数,实际上对于非单例的组件,从项目架构上来说,理论上应该是从构造函数注入进去的而不是手动解析。 需要手动解析的应该为一些配置帮助类等

对于一个具体组件(类)的生命周期分为以下几种(后面的函数是autofac对应的函数):

每个依赖一个实例(Instance Per Dependency) (默认) ----InstancePerDependency()
单一实例(Single Instance) 单例 ----SingleInstance()
每个生命周期作用域一个实例(Instance Per Lifetime Scope)----InstancePerLifetimeScope()
每个匹配的生命周期作用域一个实例(Instance Per Matching Lifetime Scope)----InstancePerMatchingLifetimeScope()
每个请求一个实例(Instance Per Request) asp.net web请求----InstancePerRequest()
每次被拥有一个实例(Instance Per Owned) ----InstancePerOwned()
如果你以前在传统的ASP.NET MVC项目中用过autofac,需要注意一些区别:

.net Core中需要使用InstancePerLifetimeScope替代之前(传统asp.net)的InstancePerRequest,保证每次HTTP请求只有唯一的依赖实例被创建。InstancePerRequest请求级别已经不存在了
.net Core中Web Api与Mvc的注册方式一样
.net Core中不再需要注册控制器,控制器由.net core创建,不归autofac管理(除了控制器的构造函数),这也解释了为什么不再使用InstancePerRequest生命周期,但是可以通过AddControllersAsServices()函数改变,想要深入了解的可以查看:https://www.strathweb.com/2016/03/the-subtle-perils-of-controller-dependency-injection-in-asp-net-core-mvc/
AutoFac 在asp .net core中的使用
在.net core 中使用autofac还是比较简单的,相比于传统的asp.net web 项目,省去了很多步骤

引入nuget程序包:

Autofac
Autofac.Extensions.DependencyInjection
startup 中代码:

public static IContainer AutofacContainer;
// This method gets called by the runtime. Use this method to add services to the container.
public IServiceProvider ConfigureServices(IServiceCollection services)
{
    //注册服务进 IServiceCollection
    services.AddMvc();
    ContainerBuilder builder = new ContainerBuilder();
    //将services中的服务填充到Autofac中.
    builder.Populate(services);
    //新模块组件注册
    builder.RegisterModule<DefaultModuleRegister>();
    //创建容器.
    AutofacContainer = builder.Build();
    //使用容器创建 AutofacServiceProvider 
    return new AutofacServiceProvider(AutofacContainer);
}

 

上面代码调用了builder的RegisterModule函数,这个函数需要传入一个TModule的泛型,称之为autofac的模块

模块的功能就是把所有相关的注册配置都放在一个类中,使代码更易于维护和配置,下面展示了DefaultModuleRegister中的代码

DefaultModuleRegister:

public class DefaultModuleRegister : Module
{
protected override void Load(ContainerBuilder builder)
{
//注册当前程序集中以“Ser”结尾的类,暴漏类实现的所有接口,生命周期为PerLifetimeScope
builder.RegisterAssemblyTypes(System.Reflection.Assembly.GetExecutingAssembly()).Where(t => t.Name.EndsWith(“Ser”)).AsImplementedInterfaces().InstancePerLifetimeScope();
builder.RegisterAssemblyTypes(System.Reflection.Assembly.GetExecutingAssembly()).Where(t => t.Name.EndsWith(“Repository”)).AsImplementedInterfaces().InstancePerLifetimeScope();
//注册所有"MyApp.Repository"程序集中的类
//builder.RegisterAssemblyTypes(GetAssembly(“MyApp.Repository”)).AsImplementedInterfaces();
}

public static Assembly GetAssembly(string assemblyName)
{
    var assembly = AssemblyLoadContext.Default.LoadFromAssemblyPath(AppContext.BaseDirectory + $"{assemblyName}.dll");
    return assembly;
}

 

}
Configure函数中可以选择性的加上程序停止时Autofac的释放函数:

public void Configure(IApplicationBuilder app, IHostingEnvironment env, IApplicationLifetime appLifetime)
{
    if (env.IsDevelopment())
    {
        app.UseDeveloperExceptionPage();
        app.UseBrowserLink();
    }
    else
    {
        app.UseExceptionHandler("/Home/Error");
    }

    app.UseStaticFiles();

    app.UseMvc(routes =>
    {
        routes.MapRoute(
            name: "default",
            template: "{controller=Home}/{action=Index}/{id?}");
    });
    //程序停止调用函数
    appLifetime.ApplicationStopped.Register(() => { AutofacContainer.Dispose(); });
}

 

Controller中代码:

private IUserSer _user;
private IUserSer _user2;
public HomeController(IUserSer user, IUserSer user2)
{
    _user = user;
    _user2 = user2;
}
public IActionResult Index()
{
    using (var scope = Startup.AutofacContainer.BeginLifetimeScope())
    {
        IConfiguration config = scope.Resolve<IConfiguration>();
        IHostingEnvironment env = scope.Resolve<IHostingEnvironment>();
    }
    string name = _user.GetName();
    string name2 = _user2.GetName();
    return View();
}

 

可以看到,因为我们将IServiceCollection中的服务填充到了autofac中了,所以现在可以在任何位置通过AutoFac解析出来.net core默认注入的服务(IConfiguration,IHostingEnvironment等)了

正常项目使用中,我们应该将AutofacContainer放在一个公共的类库中以便各个工程均可调用

 

 

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

posted on 2023-07-27 12:23  大西瓜3721  阅读(1189)  评论(0编辑  收藏  举报

导航