下面举个例子说明依赖注入使用上的场景
//一开始需求简单,没有依赖注入
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框架中都是这么做的。
点击下一页,我们将讲解依赖注入容器相关理念。