
当我入门.NET编程时,曾自信满满地以为照旧完满掌持了依赖注入(DI)。构造函数注入、服务注册、作用域与单例——我无所不知!
直到某天出产环境突发关键事故:诓骗性能骤降、内存露出、行径相等。排查历程中发现的问题,骤然击碎了我的自信泡沫。
我原以为懂DI,却从未着实通晓其精髓。
本文将揭示99%熏陶者齐不了解的DI深层常识,这些领悟将让你的代码更易小器、便于测试且具备膨大性。
依赖注入=构造函数注入?大齐东谈主对DI的通晓停留在构造函数注入:
public class OrderService
{
private readonly IEmailService _emailService;
public OrderService(IEmailService emailService)
{
_emailService = emailService;
}
public void PlaceOrder(Order order)
{
// 业务逻辑
_emailService.SendConfirmation(order);
}
}这照实是DI的一种体式。但DI远不啻于此,还包括:
• 属性注入• 步骤注入• 服务定位器形状(尽管是反形状)构造函数注入仅仅最好奉行,而非DI的一起内涵。
误区1:将Scoped服务注入Singletonservices.AddScoped;
services.AddSingleton;若将IUserContext注入ReportGenerator,运转时将抛出相等:"无法从单例消耗作用域服务"惩处决策:
• DI容器必须感知生命周期• 作用域服务经久不成注入单例类误区2:过多依赖项——天主类空洞征当类构造函数注入5个以上服务时:
public MyController(IServiceA a, IServiceB b, IServiceC c, IServiceD d, IServiceE e)这是典型的代码异味,米兰app因为:
• 难以小器• 违背单一行状原则(SRP)• 测试复杂度激增惩处决策:
• 使用门面形状• 拆分为微型服务类误区3:虽注入接口,紧耦合仍在!咱们常以为接口代表松耦合,但若接口仅有一个完了,那便是伪抽象:
public interface IEmailService
{
void Send(string to, string message);
}
public class SmtpEmailService : IEmailService
{
public void Send(string to, string message)
{
// SMTP完了
}
}问题根源:当需要切换SendGrid或MailGun时,全代码库仍依赖SmtpEmailService 惩处决策:
• 正确的接口封闭• 为不同完了使用战术形状误区4:用服务定位器疏忽DI原则public class OrderService
{
public void PlaceOrder
{
var emailService = ServiceLocator.Get;
}
}这是典型的反形状,会导致保密依赖。 正解:宝石构造函数注入
救我于水火的最好奉行• 幸免无谓要的接口抽象• 通晓生命周期不匹配问题(单例/作用域/瞬时)• 掌持组合根(Composition Root)见解• 考证构造函数依赖项• 磋商单位测试的易模拟性附赠常识:什么是组合根?组合根是荟萃注册悉数依赖的方位,往往是Program.cs或Startup.cs:
builder.Services.AddScoped;
builder.Services.AddScoped;保持组合根整洁能显耀提高可人惜性。
我曾以为DI便是创建接口并通过构造函数注入——如斯浅易。但这实质是个领悟罗网。依赖注入实则是一种玄学米兰app,一种让诓骗松耦合、易测试、高活泼性的狡计念念想。
开云kaiyun中国官网入口