17370845950

C#怎么实现依赖注入 C# DI容器配置方法
C#依赖注入核心是用IServiceCollection注册服务、ServiceProvider解析使用;支持Transient(每次新建)、Scoped(每请求共享)、Singleton(全局唯一)三种生命周期;需避免手动new或直接调用GetService。

在C#中实现依赖注入(DI),核心是用 IServiceCollection 注册服务,再通过 ServiceProvider 解析使用。.NET 6+ 默认内置了轻量级 DI 容器,无需额外引用第三方库就能完成绝大多数场景的配置。

基础服务注册方式

注册服务有三种生命周期模式,对应不同使用场景:

  • Transient(瞬时):每次请求都新建实例,适合无状态、轻量对象,如工具类、DTO映射器
  • Scoped(作用域):每个请求(HTTP上下文)内共享一个实例,常用于数据库上下文(DbContext)
  • Singleton(单例):整个应用生命周期只创建一次,适合配置管理器、缓存客户端等

示例代码:

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 则导致并发异常。按需选择,别硬套。