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("user@example.com", "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 user@example.com with subject: Hello
Message: This is a test email.
Press Enter to exit...

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

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

.net中优秀依赖注入框架Autofac看一篇就够了的更多相关文章

  1. C#实现多级子目录Zip压缩解压实例 NET4.6下的UTC时间转换 [译]ASP.NET Core Web API 中使用Oracle数据库和Dapper看这篇就够了 asp.Net Core免费开源分布式异常日志收集框架Exceptionless安装配置以及简单使用图文教程 asp.net core异步进行新增操作并且需要判断某些字段是否重复的三种解决方案 .NET Core开发日志

    C#实现多级子目录Zip压缩解压实例 参考 https://blog.csdn.net/lki_suidongdong/article/details/20942977 重点: 实现多级子目录的压缩, ...

  2. [译]ASP.NET Core Web API 中使用Oracle数据库和Dapper看这篇就够了

    [译]ASP.NET Core Web API 中使用Oracle数据库和Dapper看这篇就够了 本文首发自:博客园 文章地址: https://www.cnblogs.com/yilezhu/p/ ...

  3. 【转】 依赖注入框架Autofac的简单使用

    Autofac是一款IOC框架,比较于其他的IOC框架,如Spring.NET,Unity,Castle等等所包含的,它很轻量级性能上也是很高的.于是,今天抽空研究了下它.下载地址:http://co ...

  4. 依赖注入框架Autofac的简单使用

    http://www.cnblogs.com/liping13599168/archive/2011/07/16/2108209.html Autofac是一款IOC框架,比较于其他的IOC框架,如S ...

  5. [转]依赖注入框架Autofac的简单使用

    本文转自:http://www.nopchina.net/post/autofac.html 话说nopcommerce底层用到了autofac框架,这里转了一篇文章简单说明一下: Autofac是一 ...

  6. 依赖注入框架Autofac源码阅读指南

    官方网站http://autofac.org/ 源码下载地址https://github.com/autofac/Autofac 最新版本是3.5.0 下载后大小为37M,包括源码,示例文档,与之相关 ...

  7. .NET Core 中依赖注入框架详解 Autofac

    本文将通过演示一个Console应用程序和一个ASP.NET Core Web应用程序来说明依赖注入框架Autofac是如何使用的 Autofac相比.NET Core原生的注入方式提供了强大的功能, ...

  8. 如何在 ASP.Net Web Forms 中使用依赖注入

    依赖注入技术就是将一个对象注入到一个需要它的对象中,同时它也是控制反转的一种实现,显而易见,这样可以实现对象之间的解耦并且更方便测试和维护,依赖注入的原则早已经指出了,应用程序的高层模块不依赖于低层模 ...

  9. 类比Spring框架来实现OC中的依赖注入

    如果你之前使用过JavaEE开发中的Spring框架的话,那么你一定对依赖注入并不陌生.依赖注入(DI: Dependency Injection)是控制反转(IoC: Inversion of Co ...

  10. .net core程序中使用微软的依赖注入框架

    我之前在博文中介绍过Asp.net core下系统自带的依赖注入框架,这个依赖框架在Microsoft.Extensions.DependencyInjection中实现,本身并不是.net core ...

随机推荐

  1. composer 的使用和常用命令大全

    composer 常用命令 1.composer初始化 init 如何手动创建 composer.json 文件.实际上还有一个 init 命令可以更容易的做到这一点. 查看当前版本composer ...

  2. filter() 函数的学习

    1.    filter() 函数 用于过滤序列,过滤掉不符合条件的元素,返回一个迭代器对象,如果要转换为列表,可以使用 list() 来转换.该接收两个参数, 第一个为函数,第二个为序列,序列的每个 ...

  3. SAP 传输请求释放及传输过程 SE10 STMS

    T-CODE:SE10 STMS 1.传输请求释放 首先通过SE10打开传输组织器. 点击[显示],可以看到待释放的请求. 此时将可修改请求中的请求,点击进行展开,可以看到子请求号和请求属性. 选中请 ...

  4. Linux虚拟机报错Job for network.service failed because the control process exited with error codeLinux虚拟机报错的解决方法

    发布于 2 天前  3 次阅读 Linux虚拟机设置静态ip后,突然发现联网连不上了,ssh也无法使用,重启network后仍旧无法使用.按照网络上的方法发现没有效果后,右键如下位置将nat模式转换为 ...

  5. 组合查询(left_inner_right)与排序(order by _DESC _ASC)在题目中的应用

    1,想要让哪一列放在开头或者结尾,只需要将select中的查询位置放在最开始或者结尾即可: 2,组合查询要注意使用 on 加上组合条件: 3,order by 默认升序(ASC),降序使用:order ...

  6. QA||TypeError: ‘module‘ object is not callable报错怎么debugIHRM接口自动化测试

    unittest.py生成测试报告时执行报错:TypeError: 'module' object is not callable 代码如下 原因:结合pycharm自动标注和报错信息,分析出应该是H ...

  7. 痞子衡嵌入式:MCUBootUtility v5.3发布,利用XMCD轻松使能外部RAM

    -- 痞子衡维护的 NXP-MCUBootUtility 工具距离上一个大版本(v5.0.0)发布过去4个多月了,期间痞子衡也做过三个小版本更新,但不足以单独介绍.这一次痞子衡为大家带来了全新重要版本 ...

  8. 什么是vfs以及它的作用

    VFS(Virtual File System,虚拟文件系统)是计算机操作系统中的一个概念,它提供了一个统一的抽象层,使得操作系统可以支持不同的文件系统类型和存储设备,而不需要直接与每个文件系统进行交 ...

  9. PostgreSQL学习笔记-1.基础知识:创建、删除数据库和表格

    PostgreSQL 创建数据库 PostgreSQL 创建数据库可以用以下三种方式:1.使用 CREATE DATABASE SQL 语句来创建.2.使用 createdb 命令来创建.3.使用 p ...

  10. Windows下音视频对讲演示程序(声学回音消除、噪音抑制、语音活动检测、自动增益控制、自适应抖动缓冲)(2023年07月13日更新)

    Windows下音视频对讲演示程序 必读说明 简介   本软件根据<道德经>为核心思想而设计,实现了两个设备之间进行音视频对讲,一般可用于楼宇对讲.智能门铃对讲.企业员工对讲.智能对讲机. ...