新闻中心

C#的依赖注入(DI)是什么?面向初学者的DI核心概念与实例教程

2025-11-24
浏览次数:
返回列表
依赖注入通过外部注入对象实现解耦,提升代码可测试性与维护性。示例中UserService不再自行创建UserDataAccess,而是通过构造函数接收IUserDataAccess实现,.NET内置容器在Program.cs中注册服务生命周期(Scoped/Singleton/Transient),运行时自动注入实例,测试时可替换为Mock对象,避免硬编码依赖,统一管理对象创建,降低耦合度。

c#的依赖注入(di)是什么?面向初学者的di核心概念与实例教程

依赖注入(Dependency Injection,简称 DI)是 C# 和 .NET 开发中一个非常重要的设计模式。它帮助我们写出更松耦合、更容易测试和维护的代码。如果你刚接触这个概念,可能会觉得抽象,但其实它的核心思想很简单:不要自己创建对象,而是让外部系统把需要的对象“送进来”。

什么是依赖?

在面向对象编程中,一个类经常需要使用另一个类的功能。比如,有一个 UserService 类需要访问数据库,它可能依赖于一个 UserDataAccess 类。这种“需要别人帮忙完成工作”的关系就是“依赖”。

示例:

假设你有一个用户服务类:

public class UserService
{
    private UserDataAccess _dataAccess;
<pre class="brush:php;toolbar:false;">public UserService()
{
    _dataAccess = new UserDataAccess(); // 手动创建依赖
}

public string GetUser(int id)
{
    return _dataAccess.GetUserById(id);
}

}

这里的问题是:UserService 自己创建了 UserDataAccess,导致两者紧紧绑在一起。如果将来想换成 Mock 数据或不同的实现,就必须修改代码 —— 这不利于测试和扩展。

依赖注入的核心思想

DI 的基本思路是:把依赖项从外部“注入”进来,而不是在类内部自己 new 出来。最常见的方式是通过构造函数传入。

改写上面的例子:

public class UserService
{
    private readonly IUserDataAccess _dataAccess;
<pre class="brush:php;toolbar:false;">public UserService(IUserDataAccess dataAccess)  // 依赖通过构造函数传入
{
    _dataAccess = dataAccess;
}

public string GetUser(int id)
{
    return _dataAccess.GetUserById(id);
}

}

现在,UserService 不关心具体是谁提供数据访问功能,只要对方符合 IUserDataAccess 接口就行。这就实现了“解耦”。

.NET 中的内置 DI 容器

.NET Core 及以后版本内置了依赖注入容器,可以在程序启动时注册服务,并自动把它们注入到需要的地方。

步骤如下:

PictoGraphic PictoGraphic

AI驱动的矢量插图库和插图生成平台

PictoGraphic 133 查看详情 PictoGraphic
  • 定义接口和实现
  • Program.csStartup.cs 中注册服务
  • 在类中通过构造函数接收依赖

完整示例:

// 1. 定义接口
public interface IUserDataAccess
{
    string GetUserById(int id);
}
<p>// 2. 实现接口
public class UserDataAccess : IUserDataAccess
{
public string GetUserById(int id)
{
return $"User {id} from database";
}
}</p><p>// 3. 使用依赖的服务
public class UserService
{
private readonly IUserDataAccess _dataAccess;</p><pre class="brush:php;toolbar:false;">public UserService(IUserDataAccess dataAccess)
{
    _dataAccess = dataAccess;
}

public string GetUser(int id)
{
    return _dataAccess.GetUserById(id);
}

}

Program.cs(.NET 6+)中注册服务:

var builder = WebApplication.CreateBuilder(args);
<p>// 注册服务:告诉容器 IUserDataAccess 应该用 UserDataAccess 来实例化
builder.Services.AddScoped<IUserDataAccess, UserDataAccess>();
builder.Services.AddScoped<UserService>();</p><p>var app = builder.Build();</p><p>// 示例:在 Minimal API 中使用
app.MapGet("/user/{id}", (int id, UserService userService) =>
{
return userService.GetUser(id);
});</p><p>app.Run();

当你访问 /user/1 时,.NET 会自动创建 UserService 实例,并把已注册的 IUserDataAccess 实现注入进去。

三种常见的服务生命周期

注册服务时,你需要选择它的生命周期:

  • Scoped:每次 HTTP 请求创建一次(适用于 Web 应用)
  • Singleton:整个程序运行期间只创建一次
  • Transient:每次请求都创建新实例

一般情况下:

  • 数据访问类用 Scoped
  • 工具类或无状态服务可用 Singleton
  • 轻量级、有状态的临时对象可用 Transient

为什么使用 DI?好处有哪些?

  • 易于测试:你可以注入 Mock 对象进行单元测试
  • 解耦代码:类之间不再硬编码依赖关系
  • 便于维护:更换实现时只需修改注册代码,无需改动业务类
  • 统一管理对象创建:避免手动 new 导致的资源浪费或错误

举个测试的例子:

// 测试时可以注入一个假的数据访问层
public class MockUserDataAccess : IUserDataAccess
{
    public string GetUserById(int id) => $"Mock User {id}";
}
<p>// 单元测试中
var mockDataAccess = new MockUserDataAccess();
var userService = new UserService(mockDataAccess); // 注入模拟对象
var result = userService.GetUser(1);
Console.WriteLine(result); // 输出: Mock User 1

这样不需要真实数据库就能测试逻辑。

基本上就这些。依赖注入不是魔法,它只是一个帮你更好地组织代码的工具。刚开始可能会觉得多写了不少接口和注册代码,但随着项目变大,你会发现它带来的清晰结构和灵活性非常值得。

以上就是C#的依赖注入(DI)是什么?面向初学者的DI核心概念与实例教程的详细内容,更多请关注其它相关文章!


# 是在  # 软装网站建设  # 网站seo优化要多少钱  # 猪肉营销视频推广  # 常德抖音seo优化推广  # 2820seo怎么做  # 公众号微网站建设  # 驻马店产品推广营销怎么靠前  # 大龙网站推广哪家便宜  # 泌阳推广网站团队  # 普兰店seo关键词优化  # 当你  # 不需要  # 就能  # 你可以  # 编码  # 如果你  # 是一个  # 不安全  # 如何使用  # 面向对象  # 为什么  # .net  # 数据访问  # c#  # 面向对象编程  # 工具  # access  # app 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: 小米汽车11月交付量突破40000台!雷军:将继续努力  腾讯QQ邮箱官方网站_QQ邮箱网页版在线登录  Yandex免登录官网入口_俄罗斯Yandex搜索引擎直达链接  Golang如何优化内存分配与垃圾回收_Golang内存管理与GC优化实践  Golang如何使用net/url解析URL_Golang URL解析与处理方法  Win11怎么开启高性能模式_Windows 11电源计划优化设置  Go语言JSON解析深度指南:动态访问与结构体映射实践  如何仅使用CSS更改登录界面背景图像图标的颜色  Win11怎么查看显卡显存 Win11显示适配器属性及专用视频内存查询  UC浏览器网页版登录入口官网 电脑版网址入口  12306选座怎么选到临时改签座_12306改签选座策略与步骤  双系统安装时,如何设置默认启动系统? msconfig命令了解一下!  生成rdflib自定义SPARQL函数:参数匹配与实践指南  Promise错误处理:在catch后终止链式then执行的策略  J*aScript井字棋(Tic-Tac-Toe)核心交互逻辑实现教程  邮编格式怎么匹配地址_根据邮编格式快速匹配详细地址的技巧  免费抖音短视频入口_抖音网页版短视频免费通道  如何有效阻止外部脚本意外修改内联样式的高度属性  PHP中高效并行检查多链接状态的教程  构建轻量级网站内部消息系统:Formspree 集成指南  Yandex官网免登录入口_俄罗斯Yandex搜索引擎一键访问  抓大鹅解压小游戏 抓大鹅摸鱼解压入口  C++20的source_location是什么_C++在编译期获取源码位置信息用于日志和断言  win11如何加载ICC颜色配置文件 Win11校色文件安装与显示器色彩管理【指南】  PHP表单数据传递:如何通过隐藏输入字段获取动态ID  C#使用XPath查询节点时出错? 常见语法错误与调试技巧  WordPress插件开发:正确注册卸载钩子与避免常见陷阱  age动漫网站入口 age动漫官网直接访问入口  uc浏览器网页版入口 uc浏览器网页版最新网址  Node.js CSV 数据处理:基于字段值条件过滤整条记录的策略  蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】  提升屏幕阅读器对“m”时间单位的播报准确性:HTML与CSS组合解决方案  《主播少女的秘密账号迷宫》首支宣传片  Composer如何处理Git子模块(submodule)依赖_Composer与Git Submodule的对比与选择  单12V-2&#215;6实现为RTX 5090供电750W!甚至都没敢跑分  Tabulator表格中精确实现日期时间排序的指南  Win10如何恢复误删的快捷方式_Win10重建常用软件快捷方式  php源码怎么在电脑上测试_电脑测试php源码方法步骤【教程】  淘宝支付提示失败如何解决 淘宝支付流程优化方法  composer 和 npm/yarn 在管理依赖方面有什么核心思想差异?  红果短剧网页版官网入口 官方最新网址发布  夸克浏览器网页版最新地址 夸克浏览器官方入口合集  Django模型中自动计算可用余额的实现方法  在VS Code中配置和运行Dart程序的完整步骤  TikTok国际版网页端快速入口 TikTok全球版短视频浏览教程  Windows7怎么硬盘安装 Windows7提取ISO镜像到非系统盘并运行setup.exe实现硬盘直装【教程】  凉拌黄瓜怎么拌更入味 凉拌黄瓜简单家常做法  XML中包含HTML标签导致解析错误? 正确嵌入非XML数据的两种方法  在FastAPI中利用lifespan与依赖注入高效管理Redis连接池  印象笔记如何设离线包出差查阅_印象笔记设离线包出差查阅【离线阅读】 

搜索