下面举个例子说明依赖注入使用上的场景
//一开始需求简单,没有依赖注入 abstract class Human { } class Woman extends Human { } class Man extends Human { protected $wife; public function __construct() { $this->wife = new Woman(); } public function kissWife() { echo "the man kissed his wife"; } } $man = new Man(); $man->kissWife();
随着访问量越来越大,玩的人越来也多,新需求随之而来。。
产品经理:妻子改成可以是男的吧,好多用户有这个需求,这样玩我们游戏的肯定更多。
程序猿:擦,Wife又可以是Man,又可以是Woman,这可怎么弄?
这个时候,依赖注入就可以闪亮登场了。
abstract class Human { } class Woman extends Human { } class Man extends Human { protected $wife; public function setWife(Human $human) { $this->wife = $human; } public function kissWife() { echo "the man kissed his wife"; } } $man = new Man(); $man->setWife(new Woman());//注意这里 $man->kissWife(); $anotherMan = new Man(); $anotherMan->setWife(new Man());//注意这里 $anotherMan->kissWife();
这里我们看到,依赖注入的可以是继承依赖类的任何类,所以现在Man的Wife既可以是Woman也可以是Man。玩的人越来也多,新需求随之而来。。。
产品经理:把妻子改成伴侣吧,伴侣里面除了Man和Woman再加个Cat,好多用户有这个需求,这样玩我们游戏的肯定更多。
程序猿:擦,又是Man又是Woman还有Cat,幸好我会依赖注入。
abstract class Human { } interface canBePartner { } class Cat implements canBePartner { } class Woman extends Human implements canBePartner { } class Man extends Human implements canBePartner { protected $partner; public function setPartner(canBePartner $partner) { $this->partner = $partner; } public function kissPartner() { echo "the man kissed his partner"; } } $man = new Man(); $man->setPartner(new Woman()); $man->kissPartner(); $man2 = new Man(); $man2->setPartner(new Man()); $man2->kissPartner(); $man3 = new Man(); $man3->setPartner(new Cat()); $man3->kissPartner();
这里我们看到,依赖注入不但可以是继承依赖类的所有子类,也可以是实现依赖接口的所有类。所以如果我们在伴侣中再加入一个Dog,只需要让它实现canBePartner接口就可以了:
class Dog implements canBePartner { } $man = new Man(); $man->setPartner(new Dog());
依赖注入虽然降低了耦合度,但是也有缺点,就是需要我们自己管理注入的对象。所以,在实际应用中,我们通常需要实现一个容器去管理和实现依赖对象的注入。实际上,PHP的常用Web框架中都是这么做的。
点击下一页,我们将讲解依赖注入容器相关理念。