首页 > 其他分享 >.net中优秀依赖注入框架Autofac看一篇就够了

.net中优秀依赖注入框架Autofac看一篇就够了

时间:2023-12-03 18:44:37浏览次数:34  
标签:Autofac builder 就够 public var 注册 组件 net ContainerBuilder

 

Autofac 是一个功能丰富的 .NET 依赖注入容器,用于管理对象的生命周期、解决依赖关系以及进行属性注入。本文将详细讲解 Autofac 的使用方法,包括多种不同的注册方式,属性注入,以及如何使用多个 ContainerBuilder 来注册和合并组件。我们将提供详细的源代码示例来说明每个概念。

1. 安装 Autofac

首先,确保你已经安装了 Autofac NuGet 包。你可以使用 NuGet 包管理器或通过控制台运行以下命令来安装 Autofac:

Install-Package Autofac

2. 创建一个简单的控制台应用程序

我们将从一个简单的控制台应用程序开始,以演示 Autofac 的基本用法。我们将创建一个包含多个组件的容器,并演示多种注册方式以及属性注入的方法。

Program.cs

using System;
using Autofac;

namespace AutofacExample
{
    class Program
    {
        static void Main(string[] args)
        {
            // 步骤 1:创建 ContainerBuilder
            var builder = new ContainerBuilder();

            // 步骤 2:注册组件
            builder.RegisterType<DatabaseConnection>().As<IDatabaseConnection>().SingleInstance();
            builder.RegisterType<UserRepository>().As<IUserRepository>().InstancePerLifetimeScope();
            builder.RegisterType<Logger>().As<ILogger>().Named<ILogger>("ConsoleLogger");

            // 步骤 3:构建容器
            var container = builder.Build();

            // 步骤 4:解析组件并进行属性注入
            using (var scope = container.BeginLifetimeScope())
            {
                var userRepository = scope.Resolve<IUserRepository>();
                userRepository.AddUser("John Doe");

                // 属性注入示例
                var logger = scope.ResolveNamed<ILogger>("ConsoleLogger");
                logger.Log("This is a log message with attribute injection.");
            }

            Console.WriteLine("Press Enter to exit...");
            Console.ReadLine();
        }
    }
}

3. 创建组件和接口

现在,我们将创建三个组件 DatabaseConnection,UserRepository 和 Logger,以及它们所实现的接口。

DatabaseConnection.cs

public interface IDatabaseConnection
{
    void Connect();
}

public class DatabaseConnection : IDatabaseConnection
{
    public void Connect()
    {
        Console.WriteLine("Connected to the database.");
    }
}

UserRepository.cs

public interface IUserRepository
{
    void AddUser(string username);
}

public class UserRepository : IUserRepository
{
    private readonly IDatabaseConnection _databaseConnection;

    public UserRepository(IDatabaseConnection databaseConnection)
    {
        _databaseConnection = databaseConnection;
    }

    public void AddUser(string username)
    {
        _databaseConnection.Connect();
        Console.WriteLine($"User '{username}' added to the database.");
    }
}

Logger.cs

public interface ILogger
{
    void Log(string message);
}

public class Logger : ILogger
{
    public void Log(string message)
    {
        Console.WriteLine($"Logging: {message}");
    }
}

4. 多种注册方式

Autofac 提供了多种不同的组件注册方式,允许你控制组件的生命周期、解决复杂的依赖关系和应用更高级的用法。以下是一些常见的注册方式:

4.1. 单例注册

你可以注册一个组件为单例,这意味着容器将返回同一个实例,直到容器被销毁。在示例中,我们使用 SingleInstance() 方法将 DatabaseConnection 注册为单例。

builder.RegisterType<DatabaseConnection>().As<IDatabaseConnection>().SingleInstance();

4.2. 生命周期范围注册

你可以将组件注册为具有特定生命周期范围,例如单次请求或单个生命周期。在示例中,我们使用 InstancePerLifetimeScope() 方法将 UserRepository 注册为单个生命周期。

builder.RegisterType<UserRepository>().As<IUserRepository>().InstancePerLifetimeScope();

4.3. 命名注册

你可以注册组件并为其指定一个名称,以便在解析时根据名称来选择不同的实现。在示例中,我们使用 Named<TService, TImplementer>(string name) 方法为 Logger 注册一个名为 "ConsoleLogger" 的实现。

builder.RegisterType<Logger>().As<ILogger>().Named<ILogger>("ConsoleLogger");

4.4. Lambda 表达式注册

你可以使用 Lambda 表达式注册一个组件,以根据需要创建实例。在示例中,我们使用 Lambda 表达式注册 DatabaseConnection。

builder.Register(c => new DatabaseConnection()).As<IDatabaseConnection>();

4.5. 泛型组件注册

你可以注册泛型组件,允许你在解析时提供类型参数。在示例中,我们使用 RegisterGeneric 方法注册泛型组件 GenericRepository<T>。

builder.RegisterGeneric(typeof(GenericRepository<>)).As(typeof(IGenericRepository<>));

5. 属性注入

Autofac 允许你进行属性注入,这意味着你可以在组件实例化后注入属性的值。在示例中,我们演示了如何使用属性注入将 ILogger 注入到 UserRepository 中。

首先,我们需要为 UserRepository 类添加一个属性,并使用 [Autowired] 特性进行标记:

public class UserRepository : IUserRepository
{
    private readonly IDatabaseConnection _databaseConnection;

    // 使用 [Autowired] 特性进行属性注入
    [Autowired]
    public ILogger Logger { get; set; }

    public UserRepository(IDatabaseConnection databaseConnection)
    {
        _databaseConnection = databaseConnection;
    }

    public void AddUser(string username)
    {
        _databaseConnection.Connect();
        Console.WriteLine($"User '{username}' added to the database.");

        // 使用注入的 Logger
        Logger.Log("User added.");
    }
}

接下来,我们需要在容器构建前启用属性注入。这可以通过配置 ContainerBuilder 来实现:

var builder = new ContainerBuilder();
builder.RegisterType<DatabaseConnection>().As<IDatabaseConnection>().SingleInstance

();
builder.RegisterType<UserRepository>().As<IUserRepository>().InstancePerLifetimeScope();
builder.RegisterType<Logger>().As<ILogger>().Named<ILogger>("ConsoleLogger");

// 启用属性注入
builder.RegisterCallback(PropertyInjector.InjectProperties);

var container = builder.Build();

现在,当 UserRepository 被解析时,Logger 属性将自动注入,从而实现属性注入。

6. 使用多个ContainerBuilder合并注册

有时候,你可能需要在不同的模块或程序部分中注册组件。对于这种情况,你可以使用多个 ContainerBuilder 对象,并最终将它们合并到一个主容器中。下面是如何实现这一点的示例:

Program.cs(扩展)

在上面的示例中,我们已经创建了一个容器并注册了组件。现在,我们将添加一个额外的 ContainerBuilder,注册另一个组件,然后将它们合并。

// 步骤 7:使用另一个 ContainerBuilder 注册另一个组件
var builder2 = new ContainerBuilder();
builder2.RegisterType<EmailSender>().As<IEmailSender>();

// 步骤 8:合并 ContainerBuilder
builder.Update(builder2);

EmailSender.cs

public interface IEmailSender
{
    void SendEmail(string to, string subject, string message);
}

public class EmailSender : IEmailSender
{
    public void SendEmail(string to, string subject, string message)
    {
        Console.WriteLine($"Sending email to {to} with subject: {subject}");
        Console.WriteLine($"Message: {message}");
    }
}

现在,我们已经注册了一个名为 EmailSender 的额外组件,并将其合并到主容器中。

7. 使用多个 ContainerBuilder 示例

这是完整的示例代码:

Program.cs(完整)

using System;
using Autofac;

namespace AutofacExample
{
    class Program
    {
        static void Main(string[] args)
        {
            // 步骤 1:创建 ContainerBuilder
            var builder = new ContainerBuilder();

            // 步骤 2:注册组件
            builder.RegisterType<DatabaseConnection>().As<IDatabaseConnection>().SingleInstance();
            builder.RegisterType<UserRepository>().As<IUserRepository>().InstancePerLifetimeScope();
            builder.RegisterType<Logger>().As<ILogger>().Named<ILogger>("ConsoleLogger");

            // 步骤 3:构建容器
            var container = builder.Build();

            // 步骤 4:解析组件并进行属性注入
            using (var scope = container.BeginLifetimeScope())
            {
                var userRepository = scope.Resolve<IUserRepository>();
                userRepository.AddUser("John Doe");

                // 属性注入示例
                var logger = scope.ResolveNamed<ILogger>("ConsoleLogger");
                logger.Log("This is a log message with attribute injection.");
            }

            // 步骤 7:使用另一个 ContainerBuilder 注册另一个组件
            var builder2 = new ContainerBuilder();
            builder2.RegisterType<EmailSender>().As<IEmailSender>();

            // 步骤 8:合并 ContainerBuilder
            builder.Update(builder2);

            // 步骤 9:解析新组件
            using (var scope = container.BeginLifetimeScope())
            {
                var emailSender = scope.Resolve<IEmailSender>();
                emailSender.SendEmail("[email protected]", "Hello", "This is a test email.");
            }

            Console.WriteLine("Press Enter to exit...");
            Console.ReadLine();
        }
    }
}

这个示例演示了如何使用多个 ContainerBuilder 注册不同的组件,并将它们合并到一个容器中。当程序运行时,它会输出以下内容:

Connected to the database.
User 'John Doe' added to the database.
Logging: This is a log message with attribute injection.
Sending email to [email protected] with subject: Hello
Message: This is a test email.
Press Enter to exit...

这表明我们成功注册和合并了不同的组件,并且它们可以一起工作。

Autofac 是一个强大的 .NET 依赖注入容器,它提供了多种注册方式、属性注入以及合并多个 ContainerBuilder 的功能,使你能够更灵活地管理对象的生命周期和解决依赖关系。希望这个示例能够帮助你更好地理解 Autofac 的使用方式,并在你的.NET 项目中更好地应用依赖注入。Autofac 的强大功能使它成为一个优秀的依赖注入容器,适用于各种应用场景。

 

标签:Autofac,builder,就够,public,var,注册,组件,net,ContainerBuilder
From: https://www.cnblogs.com/hanbing81868164/p/17872593.html

相关文章

  • 全面的.NET微信网页开发之JS-SDK使用步骤、配置信息和接口请求签名生成详解
    JSSDK使用步骤步骤一:绑定安全域名:先登录微信公众平台进入“公众号设置”的“功能设置”里填写“JS接口安全域名”。步骤二:引入JS文件:在需要调用JS接口的页面引入如下JS文件,(支持https):http://res.wx.qq.com/open/js/jweixin-1.6.0.js如需进一步提升服务稳定性,当上述资源不......
  • Netty源码学习7——netty是如何发送数据的
    零丶引入系列文章目录和关于我经过《Netty源码学习4——服务端是处理新连接的&netty的reactor模式和《Netty源码学习5——服务端是如何读取数据的》,我们了解了netty服务端是如何建立连接,读取客户端数据的,通过《Netty源码学习6——netty编码解码器&粘包半包问题的解决》我们认识......
  • java.net.ConnectException: Connection timed out: connectjava.net.连接异常:连接超
    因为我有steam++加速器,但这个加速器会清空你的hosts文件,往常出现java.net.ConnectException:Connectiontimedout:connectjava.net.连接异常:连接超时:连接这个问题的时候一般都是hosts文件被清空了。但昨天写作业的时候发现在hosts文件写上ip以后还会报错。这个问题我解决了......
  • .net——管道和中间件
    .NET管道(Pipeline)和中间件(Middleware)是密切相关的两个概念,它们共同构成了.NETCore应用程序的请求处理机制。简单来说,.NET管道是一个请求处理管道,其中包含了一系列的中间件。每个中间件都负责处理请求或响应的某个方面,例如身份验证、日志记录、缓存等。当一个请求进入管道时,它会......
  • Kubernetes Pods如何访问外部域名
    Pods如何访问外网域名在Kubernetes环境中,并不是所有服务都适合部署中集群中,如数据库服务,在这种情况下集群中的Pods如何访问集群外的服务,有以下几种方式CoreDNS方式新增一段配置,如下范例范例.:53{errorshealth{lameduck15s}readykuber......
  • 【ASP.NET Core】MVC过滤器:常见用法
    前面老周给大伙伴们演示了过滤器的运行流程,大伙只需要知道下面知识点即可:1、过滤器分为授权过滤、资源访问过滤、操作方法(Action)过滤、结果过滤、异常过滤、终结点过滤。上一次咱们没有说异常过滤和终结点过滤,不过老周后面会说的。对这些过滤器,你有印象就行了。2、所有过滤器接......
  • kubernetes 安装harbor
    一、kubernetes安装harbor安装Cert-manager安装Cert-manager会自动签发免费的Let’sEncryptHTTPS证书,并在过期前自动续期。首先,运行helmrepoadd命令添加官方helm仓库#helmrepoaddjetstackhttps://charts.jetstack.io"jetstack"hasbeenaddedtoyourreposito......
  • .Net实验一 语言基础
    一、实验目的熟悉VisualStido.NET实验环境;掌握控制台程序的编写方法;掌握C#程序设计语言的语法基础;掌握控制语句和数组的使用。二、实验要求根据题目要求,编写C#程序,并将程序代码和运行结果写入实验报告。三、实验内容编写一个控制台应用程序,输入三角形或者长方形边长,计......
  • .net 下优秀的DI框架推荐,看看你用过几个?
    在.NET生态系统中,有许多出色的依赖注入(DI)框架可供选择。每个框架都有其独特的特点和优点,可以根据项目需求和偏好进行选择。下面详细介绍一些.NET中优秀的DI框架,它们的优点以及适用场景。1、Microsoft.Extensions.DependencyInjection:官方支持:Microsoft.Extensions.DependencyIn......
  • .net core Razor Page TempData不工作,RedirectToPage后无法获取值怎么办?
    问题:.netcore旧项目更新到.netcore8.0后,发现之前的错误反馈信息显示不出来了,经过反复搜索,询问人工智能无果。之前怀疑/测试过:1.新版浏览器chrome访问https://localhost是否限制了Cookie2.浏览器是否受欧盟Cookie法规的要求进行了限制。3.写法错误RazorpageTempData......