<C#> 详细介绍.NET 依赖注入

news2025/4/18 14:42:58

在 .NET 开发中,依赖注入(Dependency Injection,简称 DI)是一种设计模式,它可以增强代码的可测试性、可维护性和可扩展性。以下是对 .NET 依赖注入的详细介绍:

1. 什么是依赖注入

在软件开发里,当一个类需要使用另一个类的功能时,就会产生依赖关系。传统做法是在类的内部创建依赖对象,这样会使代码耦合度变高,不利于测试和维护。而依赖注入则是将依赖对象的创建和管理工作交给外部容器,让类只专注于自身的业务逻辑。

例如,有一个 OrderService 类需要使用 EmailService 类来发送邮件,传统方式是在 OrderService 内部创建 EmailService 实例:

public class EmailService
{
    public void SendEmail(string message)
    {
        // 发送邮件的逻辑
        Console.WriteLine($"Sending email: {message}");
    }
}

public class OrderService
{
    private readonly EmailService _emailService;

    public OrderService()
    {
        _emailService = new EmailService();
    }

    public void ProcessOrder()
    {
        // 处理订单的逻辑
        _emailService.SendEmail("Order processed successfully.");
    }
}

使用依赖注入,OrderService 不再自己创建 EmailService 实例,而是通过构造函数接收:

public class OrderService
{
    private readonly IEmailService _emailService;

    public OrderService(IEmailService emailService)
    {
        _emailService = emailService;
    }

    public void ProcessOrder()
    {
        // 处理订单的逻辑
        _emailService.SendEmail("Order processed successfully.");
    }
}

2. .NET 中的依赖注入容器

.NET 提供了内置的依赖注入容器,它能够管理对象的生命周期和依赖关系。在 Startup.cs(.NET 5 及之前版本)或者 Program.cs(.NET 6 及之后版本)中配置依赖注入。

2.1 注册服务

在 .NET 中,服务可以通过不同的生命周期进行注册,主要有以下三种:

  • 单例(Singleton):在整个应用程序生命周期中,只创建一个服务实例。
  • 作用域(Scoped):在每个请求的生命周期内,只创建一个服务实例。
  • 瞬态(Transient):每次请求服务时,都会创建一个新的服务实例。

以下是在 .NET 6 的 Program.cs 中注册服务的示例:

using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;

var host = Host.CreateDefaultBuilder(args)
    .ConfigureServices((hostContext, services) =>
    {
        // 注册单例服务
        services.AddSingleton<IEmailService, EmailService>();
        // 注册作用域服务
        services.AddScoped<IOrderService, OrderService>();
        // 注册瞬态服务
        services.AddTransient<ITransientService, TransientService>();
    })
    .Build();

// 使用服务
var orderService = host.Services.GetRequiredService<IOrderService>();
orderService.ProcessOrder();

host.Run();
2.2 解析服务

通过 IServiceProvider 接口可以从依赖注入容器中解析服务。在上面的示例中,使用 host.Services.GetRequiredService<IOrderService>() 方法获取 IOrderService 的实例。

3. 使用依赖注入

一旦服务被注册到 IServiceCollection,你可以通过构造函数注入、属性注入或方法注入来使用这些服务。

构造函数注入
public class HomeController : Controller
{
    private readonly IUserService _userService;
 
    public HomeController(IUserService userService)
    {
        _userService = userService;
    }
 
    public IActionResult Index()
    {
        var users = _userService.GetAllUsers();
        return View(users);
    }
}
属性注入
public class HomeController : Controller
{
    public IUserService UserService { get; set; }
 
    public IActionResult Index()
    {
        var users = UserService.GetAllUsers();
        return View(users);
    }
}
方法注入 (不常用)
public class HomeController : Controller
{
    public IActionResult Index([FromServices] IUserService userService)
    {
        var users = userService.GetAllUsers();
        return View(users);
    }
}
在某些情况下,你可能需要在非控制器或非组件类中访问服务。你可以通过 IServiceProvider 来实现这一点:
// 在 Startup 或 Program.cs 中构建的应用程序实例中获取服务提供者
var serviceProvider = app.Services; 

var userService = serviceProvider.GetRequiredService<IUserService>();

4. 依赖注入的优点

  • 可测试性:因为依赖对象是通过构造函数注入的,所以在单元测试时可以轻松地使用模拟对象来替代真实的依赖对象。
  • 可维护性:当依赖对象发生变化时,只需要在依赖注入容器中修改注册配置,而不需要修改使用依赖的类。
  • 可扩展性:可以方便地替换或添加新的依赖对象,而不会影响现有的代码。

5. 依赖注入的使用场景

  • Web 应用开发:在 ASP.NET Core 应用中,控制器、中间件等组件可以通过依赖注入获取所需的服务。
  • 测试驱动开发(TDD):依赖注入使得编写单元测试更加容易,因为可以使用模拟对象来隔离被测试的类。
  • 大型项目开发:在大型项目中,使用依赖注入可以有效地管理组件之间的依赖关系,降低代码的耦合度。

总结

依赖注入是 .NET 开发中非常重要的设计模式,它通过将依赖对象的创建和管理工作交给外部容器,提高了代码的可测试性、可维护性和可扩展性。.NET 提供了内置的依赖注入容器,开发者可以根据需要注册不同生命周期的服务,并从容器中解析服务实例。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2335569.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

布局决定终局:基于开源AI大模型、AI智能名片与S2B2C商城小程序的战略反推思维

摘要&#xff1a;在商业竞争日益激烈的当下&#xff0c;布局与终局预判成为企业成功的关键要素。本文探讨了布局与终局预判的智慧性&#xff0c;强调其虽无法做到百分之百准确&#xff0c;但能显著提升思考能力。终局思维作为重要战略工具&#xff0c;并非一步到位的战略部署&a…

告别循环!用Stream优雅处理集合

什么是stream&#xff1f; 也叫Stream流&#xff0c;是jdk8新增的一套API&#xff08;java.util.stream.*&#xff09;可以用于操作集合或者数组的数据。 优势&#xff1a;Stream流大量的结合了Lambda语法的风格编程&#xff0c;提供了一种更加强大&#xff0c;更加简单的方式…

Linux电源管理、功耗管理 和 发热管理 (CPUFreq、CPUIdle、RPM、thermal、睡眠 和 唤醒)

1 架构图 1.1 Linux内核电源管理的整体架构 《Linux设备驱动开发详解&#xff1a;基于最新的Linux4.0内核》图19.1 1.2 通用的低功耗软件栈 《SoC底层软件低功耗系统设计与实现》 1.3 低功耗系统的架构设计&#xff1b;图1-3 2 系统级睡眠和唤醒管理 Linux系统的待机、睡眠…

OSCP - Proving Grounds -FunboxEasy

主要知识点 弱密码路径枚举文件上传 具体步骤 首先是nmap扫描一下&#xff0c;虽然只有22&#xff0c;80和3306端口&#xff0c;但是事情没那么简单 Nmap scan report for 192.168.125.111 Host is up (0.45s latency). Not shown: 65532 closed tcp ports (reset) PORT …

Vue 3 国际化实战:支持 Element Plus 组件和语言持久化

目录 Vue 3 国际化实战&#xff1a;支持 Element Plus 组件和语言持久化实现效果&#xff1a;效果一、中英文切换效果二、本地持久化存储效果三、element Plus国际化 vue3项目国际化实现步骤第一步、安装i18n第二步、配置i18n的en和zh第三步&#xff1a;使用 vue-i18n 库来实现…

1.阿里云快速部署Dify智能应用

一、宝塔面板 宝塔面板是一款功能强大且易于使用的服务器管理软件&#xff0c;支持Linux和Windows系统&#xff0c;通过web端可视化操作&#xff0c;优化了建站流程&#xff0c;提供安全管理、计划任务、文件管理以及软件管理等功能。 1.1 宝塔面板的特点与优势 易用性 宝塔面…

在pycharm配置虚拟环境和jupyter,解决jupyter运行失败问题

记录自己pycharm环境配置和解决问题的流程。 解决pycharm无法运行jupyter代码&#xff0c;仅运行import板块显示运行失败&#xff0c;但是控制台不输出任何错误信息&#xff0c;令人困惑。 遇到的问题是&#xff1a;运行代码左下角显示运行失败但是有没有任何的输出错误信息。 …

【Docker】离线安装Docker

背景 离线安装Docker的必要性&#xff0c;第一&#xff0c;在目前数据安全升级的情况下&#xff0c;很多外网已经基本不好访问了。第二&#xff0c;如果公司有对外部署的需求&#xff0c;那么难免会存在对方只有内网的情况&#xff0c;那么我们就要做到学会离线安装。 下载安…

极简cnn-based手写数字识别程序

1.先看看识别效果&#xff1a; 这个程序识别的是0~9的一组手写数字&#xff0c;这是最终的识别效果&#xff0c;为1&#xff0c;代表识别成功&#xff0c;0为失败。 然后数据源是&#xff1a;ds deeplake.load(hub://activeloop/optical-handwritten-digits-train)里面是一组…

C++核心机制-this 指针传递与内存布局分析

示例代码 #include<iostream> using namespace std;class A { public:int a;A() {printf("A:A()的this指针&#xff1a;%p!\n", this);}void funcA() {printf("A:funcA()的this指针&#xff1a;%p!\n", this);} };class B { public:int b;B() {prin…

vue3 history路由模式刷新页面报错问题解决

在使用history路由模式时刷新网页提示404错误&#xff0c;这是改怎么办呢。 官方解决办法 https://router.vuejs.org/zh/guide/essentials/history-mode.html

PHP爬虫教程:使用cURL和Simple HTML DOM Parser

一个关于如何使用PHP的cURL和HTML解析器来创建爬虫的教程&#xff0c;特别是处理代理信息的部分。首先&#xff0c;我需要确定用户的需求是什么。可能他们想从某个网站抓取数据&#xff0c;但遇到了反爬措施&#xff0c;需要使用代理来避免被封IP。不过用户没有提到具体的目标网…

3.2.2.1 Spring Boot配置静态资源映射

在Spring Boot中配置静态资源映射&#xff0c;可以通过默认路径或自定义配置实现。默认情况下&#xff0c;Spring Boot会在classpath:/static/等目录下查找静态资源。若需自定义映射&#xff0c;可通过实现WebMvcConfigurer接口的addResourceHandlers方法或在全局配置文件中设置…

# 更换手机热点后secureCRT无法连接centOS7系统

更换手机热点后secureCRT无法连接centOS7系统 一、问题描述 某些情况下&#xff0c;我们可能使用手机共享热点而给电脑联网。本来用一个手机热点共享网络时&#xff0c;SecureCRT可以正常连接到CentOS 7虚拟机&#xff0c;当更换一个手机热点时&#xff0c;突然发现SecureCR…

jupyter notebook 无法启动- markupsafe导致

一、运行jupyter notebook和Spyder报错&#xff1a;(已安装了Anaconda&#xff0c;以前可打开) 1.背景&#xff1a;为了部署机器学习模型&#xff0c;按教程直接安装了flask 和markupsafe&#xff0c;导致jupyter notebook&#xff0c;Spyder 打不开。 pip install flas…

CTF web入门之命令执行 完整版

web29 文件名过滤 由于flag被过滤,需要进行文件名绕过,有以下几种方法: 1.通配符绕过 fla?.* 2.反斜杠绕过 fl\ag.php 3.双引号绕过 fl’‘ag’.php 还有特殊变量$1、内联执行等 此外 读取文件利用cat函数,输出利用system、passthru 、echo echo `nl flag.php`; ec…

Java 开发工具:从 Eclipse 到 IntelliJ IDEA 的进化之路

Java 开发工具&#xff1a;从 Eclipse 到 IntelliJ IDEA 的进化之路 在 Java 开发的历史长河中&#xff0c;开发工具的演变不仅改变了程序员的编码方式&#xff0c;也深刻影响了整个行业的开发效率和代码质量。从 Eclipse 到 IntelliJ IDEA&#xff0c;这不仅是工具的更替&…

GPT - 2 文本生成任务全流程

数据集下载 数据预处理 import json import pandas as pdall_data []with open("part-00018.jsonl",encoding"utf-8") as f:for line in f.readlines():data json.loads(line)all_data.append(data["text"])batch_size 10000for i in ran…

红宝书第四十三讲:基于资料的数据可视化工具简单介绍:D3.js 与 Canvas绘图

红宝书第四十三讲&#xff1a;基于资料的数据可视化工具简单介绍&#xff1a;D3.js 与 Canvas绘图12 资料取自《JavaScript高级程序设计&#xff08;第5版&#xff09;》。 查看总目录&#xff1a;红宝书学习大纲 一、D3.js&#xff1a;数据驱动文档的王者 1 核心特性&#x…

深入理解 Vue 的数据代理机制

何为数据代理&#xff1f; 通过一个对象代理对另一个对象中的属性的操作&#xff08;读/写&#xff09;&#xff0c;就是数据代理。 要搞懂Vue数据代理这个概念&#xff0c;那我们就要从Object.defineProperty()入手 Object.defineProperty()是Vue中比较底层的一个方法&…