C#依赖注入核心是用IServiceCollection注册服务、ServiceProvider解析使用;支持Transient(每次新建)、Scoped(每请求共享)、Singleton(全局唯一)三种生命周期;需避免手动new或直接调用GetService。
在C#中实现依赖注入(DI),核心是用 IServiceCollection 注册服务,再通过 ServiceProvider 解析使用。.NET 6+ 默认内置了轻量级 DI 容器,无需额外引用第三方库就能完成绝大多数场景的配置。
注册服务有三种生命周期模式,对应不同使用场景:
示例代码:
var builder = WebApplication.CreateBuilder(args); // 接口 IEmailService → 实现类 SmtpEmailService builder.Services.AddTransient(); builder.Services.AddScoped(); // 默认构造函数自动解析 builder.Services.AddSingleton ();
当构造函数需要运行时参数,或需自定义初始化逻辑时,用工厂委托注册:
// 传入配置值 builder.Services.AddSingleton(sp => new JsonConfigReader("appsettings.json")); // 基于其他已注册服务构建 builder.Services.AddScoped (sp => { var logger = sp.GetRequiredService >(); var db = sp.GetRequiredService(); return new OrderService(logger, db); });
只要类构造函数参数类型已在 DI 容器中注册,框架会自动注入:
public class OrdersController : ControllerBase { private readonly IOrderService _orderService; private readonly ILogger
_logger; public OrdersController(IOrderService orderService, ILogger logger) { _orderService = orderService; _logger = logger; } [HttpGet] public async Task Get() => Ok(await _orderService.GetAll()); }
注意:不要手动 new 对象,也不要直接调用 ServiceProvider.GetService() —— 这会破坏可测试性和生命周期管理。
把重复的注册逻辑抽成扩展方法,提高可读性和复用性:
public static class ServiceCollectionExtensions
{
public static IServiceCollection AddEmailServices(this IServiceCollection services)
{
services.AddTransient();
services.AddSingleton();
return services;
}
}
// 使用时一行搞定
builder.Services.AddEmailServices();
基本上就这些。不复杂但容易忽略的是生命周期匹配——比如把 DbContext 注成 Transient 可能引发连接泄漏,而误设为 Singleton 则导致并发异常。按需选择,别硬套。