C#教程

关注公众号 jb51net

关闭
首页 > 软件编程 > C#教程 > C# 依赖注入

C#中的依赖注入Dependency Injection DI的实现步骤

作者:CoderIsArt

在C#中依赖注入是一种实现控制反转的设计模式,用于解耦组件依赖关系,下面就来介绍一下依赖注入的实现步骤,具有一定的参考价值,感兴趣的可以了解一下

在C#中,依赖注入(Dependency Injection, DI)是一种实现控制反转(IoC)的设计模式,用于解耦组件依赖关系。以下是基于 .NET Core/ .NET 5+ 的依赖注入配置详细步骤和用法:

1. 基本概念

2. 配置依赖注入的步骤

步骤1:定义服务接口和实现类

// 定义接口
public interface IMyService
{
    string GetMessage();
}

// 实现接口
public class MyService : IMyService
{
    public string GetMessage() => "Hello from MyService!";
}

步骤2:注册服务到容器

在 Program.cs 或 Startup.cs(.NET 6+ 通常使用 Program.cs):

var builder = WebApplication.CreateBuilder(args);

// 注册服务(生命周期可选)
builder.Services.AddTransient<IMyService, MyService>();      // 瞬时
builder.Services.AddScoped<IMyService, MyService>();         // 作用域
builder.Services.AddSingleton<IMyService, MyService>();      // 单例

// 直接注册类(无接口)
builder.Services.AddTransient<MyService>();

步骤3:注入服务到目标类

通过构造函数注入:

public class MyController : ControllerBase
{
    private readonly IMyService _service;

    // 构造函数自动注入
    public MyController(IMyService service)
    {
        _service = service;
    }

    [HttpGet]
    public IActionResult Get()
    {
        return Ok(_service.GetMessage());
    }
}

3. 在非Web应用(如WPF/控制台)中使用DI

WPF 示例

安装NuGet包

Install-Package Microsoft.Extensions.DependencyInjection

配置DI容器(在 App.xaml.cs):

public partial class App : Application
{
    public IServiceProvider ServiceProvider { get; private set; }

    protected override void OnStartup(StartupEventArgs e)
    {
        var services = new ServiceCollection();
        
        // 注册服务
        services.AddTransient<IMyService, MyService>();
        services.AddSingleton<MainWindow>(); // 注册主窗口

        ServiceProvider = services.BuildServiceProvider();
        
        // 解析主窗口并显示
        var mainWindow = ServiceProvider.GetRequiredService<MainWindow>();
        mainWindow.Show();
    }
}

在窗口/ViewModel中注入

public partial class MainWindow : Window
{
    public MainWindow(IMyService service)
    {
        InitializeComponent();
        var message = service.GetMessage(); // 使用注入的服务
    }
}

4. 高级用法

4.1 注入多个实现(命名或策略模式)

// 注册多个实现
builder.Services.AddTransient<IMyService, ServiceA>();
builder.Services.AddTransient<IMyService, ServiceB>();

// 通过 IEnumerable<T> 获取所有实现
public class Consumer
{
    public Consumer(IEnumerable<IMyService> services)
    {
        foreach (var service in services)
        {
            service.DoSomething();
        }
    }
}

4.2 使用工厂模式动态创建服务

builder.Services.AddTransient<IMyService>(provider => 
{
    var config = provider.GetRequiredService<IConfiguration>();
    return new MyService(config.GetValue<string>("Mode"));
});

4.3 生命周期管理注意事项

5. 常见问题解决

错误:未注册服务

System.InvalidOperationException: No service for type 'IMyService' has been registered.

解决:检查服务是否在 IServiceCollection 中正确注册。

循环依赖
重构代码以避免构造函数循环引用,或用 Lazy<T> 延迟初始化。

6. 总结表格

操作代码示例说明
注册Transient服务services.AddTransient<IMyService, MyService>();每次请求新实例
注册Scoped服务services.AddScoped<IMyService, MyService>();同一作用域内共享实例
注册Singleton服务services.AddSingleton<IMyService, MyService>();全局单例
构造函数注入public MyClass(IMyService service)自动解析依赖
手动解析服务var service = provider.GetRequiredService<IMyService>();从容器获取实例

通过以上步骤,可以灵活地在C#项目中配置和使用依赖注入,提升代码的可测试性和可维护性。

到此这篇关于C#中的依赖注入Dependency Injection DI的实现步骤的文章就介绍到这了,更多相关C# 依赖注入内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文