0

0

c# 在高并发下,如何选择合适的 DI 生命周期(Singleton, Scoped, Transient)

幻夢星雲

幻夢星雲

发布时间:2026-01-08 05:27:28

|

169人浏览过

|

来源于php中文网

原创

Singleton 必须线程安全,避免可变状态引发竞态;Scoped 绑定请求而非线程,禁用于 HostedService 或注入 Singleton;Transient 适合无状态轻量类,禁用 HttpClient 等资源型类型;禁止短周期服务注入长周期服务。

c# 在高并发下,如何选择合适的 di 生命周期(singleton, scoped, transient)

Singleton 实例在高并发下必须线程安全

Singleton 服务在整个应用生命周期内只创建一次,所有请求共享同一个实例。这意味着如果你的类里有可变状态(比如 private List<string> _cache),多个线程同时写入就会触发竞态条件,轻则数据错乱,重则抛出 InvalidOperationExceptionNullReferenceException

常见错误现象:缓存数据突然消失、计数器值跳变、日志输出重复或缺失。

  • 确保所有字段只读(readonly)或使用线程安全类型(如 ConcurrentDictionary<TKey, TValue>ConcurrentQueue<T>
  • 避免在 Singleton 中持有 HttpContextDbContext 或任何与请求上下文强绑定的对象
  • 如果必须做状态管理,优先用外部存储(Redis、内存队列)而非类内字段

Scoped 服务不是“每个线程一个”,而是“每个请求一个”

Scoped 生命周期由 DI 容器根据当前作用域决定,在 ASP.NET Core 中默认绑定到 HttpRequest。它**不等于线程局部存储(TLS)**,也不随线程切换而隔离——异步操作中若跨 await 后仍在同一请求上下文中,仍会拿到同一个 Scoped 实例;但若在后台线程(如 Task.Run)中手动创建新作用域,则会得到新实例。

容易踩的坑:

  • HostedService 或定时任务中直接注入 Scoped 服务 → 抛出 InvalidOperationException: Cannot resolve scoped service...
  • IServiceScopeFactory.CreateScope() 手动创建作用域后忘记调用 scope.Dispose() → 导致 DbContext 连接泄漏、内存缓慢增长
  • 把 Scoped 服务注入到 Singleton 类构造函数中 → 容器启动失败,报错 Cannot consume scoped service...

Transient 适合无状态、轻量、依赖隔离强的类型

Transient 每次请求都新建实例,开销小但不为零。高频调用下(如每毫秒调用几十次的工具类),对象分配+GC 压力会上升,尤其当类内部持有大数组或未释放的非托管资源时。

适用场景:

腾讯交互翻译
腾讯交互翻译

腾讯AI Lab发布的一款AI辅助翻译产品

下载
  • DTO 映射器(如 IMapper 的自定义转换器)
  • 纯函数式工具类(无字段、只暴露静态方法的类别走 DI,但若已注册为 Transient,也无妨)
  • 需要严格隔离状态的处理器(例如每个请求都要独立初始化加密上下文)

反例:把 HttpClient 注册为 Transient —— 会导致端口耗尽。应改为 Singleton + IHttpClientFactory 管理。

混合生命周期组合要检查依赖图是否越级引用

DI 容器禁止将短生命周期服务注入长生命周期服务,例如:把 Scoped 服务注入 Singleton 构造函数。编译期不报错,运行时容器构建阶段就失败,错误信息类似:

System.InvalidOperationException: Cannot consume scoped service 'MyApp.Services.IUserService' from singleton 'MyApp.Services.INotificationService'.

排查建议:

  • dotnet trace 或第三方库(如 Scrutor)扫描注册关系
  • Program.cs 中启用验证:services.AddControllers().AddControllersAsServices(); + hostBuilder.UseDefaultServiceProvider(... ValidateOnBuild = true)
  • 对复杂依赖链,优先把状态上提到外层(如用 AsyncLocal<T> 存请求 ID),而非靠 Scoped 传递上下文

真正麻烦的是隐式依赖:某个 Transient 类内部 new 出了一个 Scoped 类,或者通过反射加载了未注册的类型——这种不会被容器校验,只能靠压测暴露。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1030

2023.08.02

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

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

765

2023.08.10

常用的数据库软件
常用的数据库软件

常用的数据库软件有MySQL、Oracle、SQL Server、PostgreSQL、MongoDB、Redis、Cassandra、Hadoop、Spark和Amazon DynamoDB。更多关于数据库软件的内容详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1006

2023.11.02

内存数据库有哪些
内存数据库有哪些

内存数据库有Redis、Memcached、Apache Ignite、VoltDB、TimesTen、H2 Database、Aerospike、Oracle TimesTen In-Memory Database、SAP HANA和ache Cassandra。更多关于内存数据库相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

671

2023.11.14

mongodb和redis哪个读取速度快
mongodb和redis哪个读取速度快

redis 的读取速度比 mongodb 更快。原因包括:1. redis 使用简单的键值存储,而 mongodb 存储 json 格式的数据,需要解析和反序列化。2. redis 使用哈希表快速查找数据,而 mongodb 使用 b-tree 索引。因此,redis 在需要高性能读取操作的应用程序中是一个更好的选择。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

501

2024.04.02

redis怎么做缓存服务器
redis怎么做缓存服务器

redis 作为缓存服务器的答案:redis 是一款开源、高性能、分布式的键值存储,可作为缓存服务器使用。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

413

2024.04.07

redis怎么解决数据一致性
redis怎么解决数据一致性

redis 提供了两种一致性模型,以维护副本数据一致性:强一致性 (sync) 确保写操作仅在复制到所有从节点后才完成;最终一致性 (async) 则在主节点上写操作后认为已完成,牺牲一致性换取性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

408

2024.04.07

mysql和redis怎么保证双写一致性
mysql和redis怎么保证双写一致性

确保 mysql 和 redis 双写一致性的技术包括:1、事务性更新:同时更新 mysql 和 redis,保证一致性;2、主从复制:mysql 主服务器更改同步到 redis 从服务器;3、基于事件的更新:mysql 记录更改并发送到 redis等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

484

2024.04.07

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 7.1万人学习

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

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