C# 容器实例生命周期

在 C#(特别是结合 .NET 的依赖注入容器,如 ASP.NET Core 内置的 DI 容器)中,服务(或组件)的生命周期主要有三种:Transient(瞬态)Scoped(作用域)Singleton(单例)。这些生命周期决定了服务实例何时被创建、重用以及销毁。

以下是它们的区别和适用场景:


1. Transient(瞬态)

  • 每次请求都创建一个新实例
  • 适用于轻量级、无状态的服务。
  • 优点:隔离性好,不会因状态共享引发问题。
  • 缺点:频繁创建/销毁可能影响性能(尤其对重量级对象)。
csharp 复制代码
services.AddTransient<IMyService, MyService>();

每次从 DI 容器解析 IMyService 时,都会返回一个全新的 MyService 实例。


2. Scoped(作用域)

  • 在同一个作用域内共享同一个实例,不同作用域使用不同实例。
  • ASP.NET Core 中,一个 HTTP 请求就是一个作用域
  • 适用于需要在一次请求中共享状态的服务(如数据库上下文 DbContext)。
csharp 复制代码
services.AddScoped<IMyService, MyService>();

在同一个 HTTP 请求中多次解析 IMyService,会得到同一个实例;但在不同请求中会得到不同实例。

⚠️ 注意:在非 Web 应用(如控制台程序)中,需要手动创建作用域(通过 IServiceScopeFactory)才能体现 Scoped 行为。


3. Singleton(单例)

  • 整个应用程序生命周期内只创建一个实例,所有请求共享该实例。
  • 适用于全局共享、无状态或线程安全的服务。
  • 优点:节省资源,避免重复初始化。
  • 风险:如果服务持有可变状态,需确保线程安全。
csharp 复制代码
services.AddSingleton<IMyService, MyService>();

整个应用运行期间,无论多少次解析 IMyService,都返回同一个实例。


对比总结

生命周期 创建时机 实例数量 适用场景
Transient 每次请求时 每次都新实例 轻量、无状态、临时对象
Scoped 每个作用域首次请求时 每个作用域一个实例 Web 请求内共享(如 DbContext)
Singleton 首次请求时(或启动时) 全局唯一实例 全局配置、缓存、日志等

补充说明

  • 不能从短生命周期服务注入长生命周期服务(例如:Singleton 服务中注入 Scoped 服务是危险的,因为 Scoped 服务可能依赖于请求上下文,而 Singleton 存活时间更长,会导致"捕获作用域外服务"的异常)。
  • 反之,长生命周期可以安全地注入短生命周期服务(但通常不推荐,因为短生命周期服务可能无法按预期工作)。

示例(ASP.NET Core)

csharp 复制代码
// Program.cs 或 Startup.cs
builder.Services.AddTransient<IOperationTransient, Operation>();
builder.Services.AddScoped<IOperationScoped, Operation>();
builder.Services.AddSingleton<IOperationSingleton, Operation>();

配合一个测试控制器,你可以观察到 GUID(代表实例 ID)的变化情况,验证不同生命周期的行为。


如有使用第三方容器(如 Autofac、Unity 等),概念类似,但 API 可能略有不同。不过 ASP.NET Core 默认 DI 容器已能满足大多数场景。

相关推荐
在放️2 小时前
Python 爬虫 · 第三方代理接入与合规使用
开发语言·爬虫·python
KANGBboy2 小时前
java知识五(继承)
java·开发语言
c++之路3 小时前
Bazel C++ 构建系列文档(三):构建第一个 C++ 项目
开发语言·c++
AI人工智能+电脑小能手3 小时前
【大白话说Java面试题 第117题】【并发篇】第17题:线程有几种状态,之间如何转换?
java·开发语言·面试
聚名网4 小时前
域名net,com,cn有区别吗?有哪些不同呢?
服务器·开发语言·php
牛油果子哥q4 小时前
STL set与map底层精讲,红黑树适配原理、有序去重特性、迭代器遍历、API实战与面试核心考点全解
开发语言·数据结构·c++·面试
foundbug9994 小时前
直流电机 PID 速度控制 MATLAB 仿真程序
开发语言·matlab
Tian_Hang5 小时前
C++原型模式(Protype)
开发语言·c++·算法
天天讯通5 小时前
OKCC 呼叫中心安全性能全解析:技术防护与管理措施指南
大数据·开发语言·网络·人工智能·安全·语音识别
xufengzhu5 小时前
第三方 Python 库 redis-py + hiredis 的使用
开发语言·redis·python