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. 论文解读(KD-UDA)《Joint Progressive Knowledge Distillation and Unsupervised Domain Adaptation》

    Note:[ wechat:Y466551 | 可加勿骚扰,付费咨询 ] 论文信息 论文标题:Joint Progressive Knowledge Distillation and Unsuperv ...

  2. SpringBoot 测试实践 - 3:@MockBean、@SpyBean 、提升测试运行速度、Testcontainer

    编写测试的时候,我们必须保证外部依赖行为一致,也需要模拟一些边界条件,所以我们需要使用 Mock 来模拟对象的行为.SpringBoot 提供了 @MockBean 和 @SpyBean 注解,可以方 ...

  3. Linux 主机磁盘繁忙度监控实战shell脚本

    Linux 磁盘繁忙度是指磁盘的使用率和活动水平.可以通过一些工具来监测磁盘繁忙度,如 iostat.iotop.sar 等. 其中,iostat 是一个常用的工具,可以提供关于磁盘活动的详细统计信息 ...

  4. Go代码包与引入:如何有效组织您的项目

    本文深入探讨了Go语言中的代码包和包引入机制,从基础概念到高级应用一一剖析.文章详细讲解了如何创建.组织和管理代码包,以及包引入的多种使用场景和最佳实践.通过阅读本文,开发者将获得全面而深入的理解,进 ...

  5. 【krpano】多分类缩略图及多分类地图案例

    该案例提供了场景多分类缩略图展示以及多地图展示,效果如下截图:                 下载地址:http://pan.baidu.com/s/1hsA5ta8 感谢群内小伙伴H·T·T的分享 ...

  6. dedebiz实时时间调用

    {dede:tagname runphp='yes'}@me = date("Y-m-d H:i:s", time());{/dede:tagname}

  7. 【最佳实践】MongoDB导入数据时重建索引

    MongoDB一个广为诟病的问题是,大量数据resotore时索引重建非常缓慢,实测5000万的集合如果有3个以上的索引需要恢复,几乎没法成功,而且resotore时如果选择创建索引也会存在索引不生效 ...

  8. CF1746F Kazaee

    prologue 数组范围一定要看好了开,不然容易我一样,调试调了一页多. 还有就是不要傻乎乎地只跑一次和哈希,因为和哈希(从下面地佬的题解中才知道)它其实算作是一种 trick(类比SA(Stimu ...

  9. 铅华洗尽,粉黛不施,人工智能AI基于ProPainter技术去除图片以及视频水印(Python3.10)

    视频以及图片修复技术是一项具有挑战性的AI视觉任务,它涉及在视频或者图片序列中填补缺失或损坏的区域,同时保持空间和时间的连贯性.该技术在视频补全.对象移除.视频恢复等领域有广泛应用.近年来,两种突出的 ...

  10. 《最新出炉》系列初窥篇-Python+Playwright自动化测试-20-处理鼠标拖拽-下篇

    1.简介 上一篇中,宏哥说的宏哥在最后提到网站的反爬虫机制,那么宏哥在自己本地做一个网页,没有那个反爬虫的机制,谷歌浏览器是不是就可以验证成功了,宏哥就想验证一下自己想法,其次有人私信宏哥说是有那种类 ...