0

0

C# 依赖注入作用域方法 C# Singleton、Scoped和Transient的区别

畫卷琴夢

畫卷琴夢

发布时间:2026-01-26 12:39:08

|

980人浏览过

|

来源于php中文网

原创

Singleton 实例在容器生命周期内只创建一次,首次请求时生成并全程复用;Scoped 按作用域(如每个 HTTP 请求)创建独立实例;Transient 每次请求都新建对象。

c# 依赖注入作用域方法 c# singleton、scoped和transient的区别

Singleton 实例在容器生命周期内只创建一次

当你注册为 Singleton,DI 容器会在第一次请求该服务时创建实例,并一直复用它——哪怕跨多个 HTTP 请求、跨线程、跨作用域。这意味着所有地方拿到的都是同一个对象引用。

适合无状态工具类(如日志记录器 ILogger)、配置读取器、或需要全局共享状态的组件(比如缓存管理器)。但要注意:如果它内部持有可变状态且没做线程同步,多线程下容易出错。

  • 注册方式:services.AddSingleton()
  • 不推荐用于依赖 Scoped 服务(如 EF Core 的 DbContext),否则会引发“Cannot resolve scoped service from root provider”错误
  • 构造函数注入的 ScopedTransient 服务,在 Singleton 中只会被解析一次(即“快照式绑定”)

Scoped 实例按作用域边界创建,常见于 Web 请求生命周期

ASP.NET Core 默认每个 HTTP 请求就是一个 Scoped 作用域。同一请求内多次 GetRequiredService() 拿到的是同一个实例;不同请求之间则各自独立。

这是 EF Core 的 DbContext 默认注册方式的原因:保证一个请求内数据库操作共享上下文,又避免跨请求状态污染。

ImgGood
ImgGood

免费在线AI照片编辑器

下载
  • 注册方式:services.AddScoped()
  • 在非托管作用域(如后台任务、Task.Run)中直接从根容器解析 Scoped 服务会失败,报错:Cannot resolve scoped service from root provider
  • 若需在后台任务中使用,必须手动创建作用域:using var scope = app.Services.CreateScope(); scope.ServiceProvider.GetRequiredService()

Transient 每次请求都新建实例,无共享状态

Transient 是最轻量的生命周期,每次调用 GetRequiredService 或构造函数注入时都会 new 一个新对象。没有复用,也没有隐式状态传递风险。

适合无状态、开销小、或需要隔离数据的类型,比如 DTO 映射器、计算工具类、或单元测试中的模拟对象。

  • 注册方式:services.AddTransient()
  • 性能上比 ScopedSingleton 略低(对象分配 + GC 压力),但多数场景可忽略
  • 可以安全注入到任何生命周期的服务中(包括 Singleton),但要注意:如果 Singleton 持有 Transient 实例,那个实例就变成“伪单例”了——它不会更新,也不会重新创建

DI 容器解析失败时的典型错误信息和排查点

最常见的报错是 InvalidOperationException: Cannot resolve scoped service 'MyApp.IDbContext' from root provider.,本质是试图在没有作用域的上下文中(如静态方法、HostedService 构造函数)直接解析 Scoped 服务。

  • 检查调用:是否在 IHostedService.StartAsyncProgram.cs 顶层代码、或静态工厂方法里用了 app.Services.GetService()
  • 确认服务注册顺序:后注册的覆盖先注册的,但生命周期不能降级(比如先 AddScopedAddSingleton 会生效后者)
  • 调试技巧:在服务构造函数里加日志或断点,观察调用次数和线程 ID,能快速识别生命周期是否符合预期
实际项目里最容易被忽略的,是 Singleton 类型中悄悄持有了 Scoped 服务的引用——它不会编译报错,但运行时可能因上下文已释放而抛出 ObjectDisposedException

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

395

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

502

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

144

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

6

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

11

2026.01.21

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

356

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2078

2023.08.14

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

31

2026.01.26

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Java 教程
Java 教程

共578课时 | 51.4万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

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

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