依赖注入本质是将对象依赖关系从内部创建改为外部传入,提升解耦、可测性与可替换性;常用构造函数注入,辅以setter注入,接口注入在php中基本不用;简易DI容器核心为绑定、解析与缓存。

依赖注入(DI)本质是把对象的依赖关系从内部创建改为外部传入,让类更专注自身职责,也更容易测试和替换实现。
为什么需要依赖注入
传统写法中,类自己 new 依赖对象,导致耦合度高、难以替换实现、不方便单元测试。比如一个订单服务依赖支付网关,如果硬编码 new Alipay(),换成 WechatPay 就得改代码。依赖注入把“谁来创建”和“谁来使用”分开,运行时由容器决定注入哪个实例。
三种注入方式怎么选
构造函数注入最常用,适合必填依赖;setter 注入适合可选或后期可变的依赖;接口注入用得少,PHP 一般不推荐。实际项目中优先用构造函数注入,语义清晰且能保证依赖不为空。
- 构造注入:在 __construct() 中接收参数,赋值给属性
- Setter 注入:提供 public setXxx() 方法,在外部调用传入依赖
- 接口注入:依赖类实现特定接口,容器通过接口识别并注入 —— PHP 里基本不用
手写一个极简 DI 容器
核心就三件事:绑定(bind)、解析(make)、缓存(单例支持)。不需要 composer 或复杂反射,几行就能跑起来:
立即学习“PHP免费学习笔记(深入)”;
// 示例:简单容器类
class Container {
private $bindings = [];
private $instances = [];
public function bind($abstract, $concrete = NULL) {
$this->bindings[$abstract] = $concrete ?: $abstract;
}
public function make($abstract) {
if (isset($this->instances[$abstract])) {
return $this->instances[$abstract];
}
$concrete = $this->bindings[$abstract] ?? $abstract;
$obj = new $concrete();
$this->instances[$abstract] = $obj;
return $obj;
}
}
绑定与解析的实际用法
先 bind 接口或抽象类到具体实现,再 make 获取实例。这样后续换实现只需改 bind 行,业务代码完全不动。
- 绑定接口:$container->bind(‘Paymentgateway’, ‘WechatPay’);
- 绑定闭包:$container->bind(‘Logger’, function() { return new FileLogger(‘/var/log/app.log’); });
- 解析使用:$orderService = new OrderService($container->make(‘PaymentGateway’));
基本上就这些。不复杂但容易忽略的是:别在容器里做太多逻辑,它只管“给什么、给谁”,具体怎么初始化交给绑定时定义。