0

0

c# 如何实现依赖注入容器

畫卷琴夢

畫卷琴夢

发布时间:2025-12-27 11:35:14

|

747人浏览过

|

来源于php中文网

原创

.NET 内置依赖注入容器是 IServiceCollection + IServiceProvider,原生支持、无需额外安装,仅提供构造函数注入和 Transient/Scoped/Singleton 生命周期管理。

c# 如何实现依赖注入容器

什么是 .NET 内置的依赖注入容器 .NET 从 Core 开始就内置了轻量级 DI 容器,类型是 IServiceCollection + IServiceProvider,它不是第三方库(比如 Autofac 或 DryIoc),而是框架原生支持的抽象层。你不需要额外安装包(Microsoft.Extensions.DependencyInjectionMicrosoft.NET.Sdk 项目中默认引用)。

它的定位是“够用、安全、与 Host 集成好”,不追求高级特性(如属性注入、装饰器、命名注册)。如果你只需要构造函数注入 + 生命周期管理(Transient / Scoped / Singleton),它完全胜任。

如何注册服务并获取实例 注册靠 IServiceCollection 扩展方法,解析靠 IServiceProviderGetService()GetRequiredService()
  • AddTransient():每次请求都新建实例
  • AddScoped():每个作用域(如一次 HTTP 请求)内复用同一个实例
  • AddSingleton():整个应用生命周期只创建一次
var services = new ServiceCollection();
services.AddTransient();
services.AddScoped();
services.AddSingleton();

var provider = services.BuildServiceProvider();

// 使用 var userService = provider.GetRequiredService();

注意:GetService() 返回 null 表示未注册;GetRequiredService() 抛出 InvalidOperationException(更推荐,避免空引用)。

为什么在 Program.cs 中用 CreateHostBuilder 而不是手动 new ServiceCollection 在 Web 项目(.NET 6+)中,你不该手动构建 IServiceProvider,而应通过 Host.CreateDefaultBuilder()WebApplicationBuilder 注入服务:
var builder = WebApplication.CreateBuilder(args);

// 注册服务(自动使用内置容器) builder.Services.AddTransient(); builder.Services.AddHttpClient();

var app = builder.Build();

原因:

星火作家大神
星火作家大神

星火作家大神是一款面向作家的AI写作工具

下载
  • 手动 BuildServiceProvider() 会提前触发容器构建,导致中间件、配置、日志等尚未注入
  • Host 会帮你处理作用域生命周期(比如 HttpContext 绑定 Scoped 服务)
  • 第三方扩展(如 EF Core、MediatR)依赖 Host 的服务发现机制

容易踩的坑:循环依赖和泛型注册 循环依赖在编译期不报错,运行时首次解析时抛出 InvalidOperationException: A circular dependency was detected。典型场景是 A 依赖 B,B 又依赖 A(包括间接依赖)。

泛型注册要注意闭合类型 vs 开放泛型:

  • services.AddTransient(typeof(IHandler ✅ 支持开放泛型映射
  • services.AddTransient, OrderCreatedHandler>(); ✅ 显式注册闭合类型
  • services.AddTransient, Handler>() ❌ 编译失败 —— C# 不允许泛型类型参数在泛型方法调用中作为实参直接出现

复杂对象图或需要 AOP/装饰器时,内置容器能力很快见顶。这时候才该考虑引入 Autofac 并替换默认容器 —— 但先确认是不是真需要,别一上来就上重武器。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

175

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

147

2025.12.18

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

226

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

430

2024.03.01

JavaScript中的typeof用法
JavaScript中的typeof用法

在JavaScript中,typeof是一个用来确定给定变量的数据类型的操作符。可以用来确定一个变量是字符串、数字、布尔值、函数、对象或undefined的数据类型。更多关于typeof用法相关文章,详情请看本专题下面的文章,php中文网欢迎大家前来学习。

738

2023.11.23

http500解决方法
http500解决方法

http500解决方法有检查服务器日志、检查代码错误、检查服务器配置、检查文件和目录权限、检查资源不足、更新软件版本、重启服务器或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

250

2023.11.09

http请求415错误怎么解决
http请求415错误怎么解决

解决方法:1、检查请求头中的Content-Type;2、检查请求体中的数据格式;3、使用适当的编码格式;4、使用适当的请求方法;5、检查服务器端的支持情况。更多http请求415错误怎么解决的相关内容,可以阅读下面的文章。

382

2023.11.14

HTTP 503错误解决方法
HTTP 503错误解决方法

HTTP 503错误表示服务器暂时无法处理请求。想了解更多http错误代码的相关内容,可以阅读本专题下面的文章。

866

2024.03.12

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

27

2025.12.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号