初识依赖注入

2017-01-09  本文已影响36人  kylesean

依赖注入,又称为控制反转(Inversion of Control,英文缩写为IoC)是一个重要的面向对象编程的法则来削减计算机程序的耦合问题,也是轻量级的Spring框架的核心。 控制反转一般分为两种类型,依赖注入(Dependency Injection,简称DI)和依赖查找(Dependency Lookup)。依赖注入应用比较广泛。IoC 可以认为是一种全新的设计模式

以上摘自百度百科,概念我们都知道是抽象的,难以理解的,要多在实际应用去慢慢体会。举个例子:

假如让我们组建一支足球队代表国家去参加世界杯,全球有这么多不同的国家,大家都想要参加比赛,那么首先,国家都有一个足球队是不是,所以 国家依赖足球队(团队),但是团队是由球员组成的是不是,所以团队依赖球员,有了这样的关系,我们就可以建立一些遵循这些关系的类,很简单,核心的三个类不能少,国家(Country),团队(Team),球员(Player),看代码:

class Country {}

class Player {}

class Team {}

这些类目前都是“空”的。让我们想一想, 首先一个 Country 得有个基本属性 name 吧,比如就交 PRC,它依赖一个 Team

以对象作为参数传递

class Country 
{
    protected $team;
    protected $name;
    
    public function __construct(Team $team, $name = 'PRC')
    {
        $this->team = $team;
        $this->name = $name;
    }
}

这里,我们传递了两个参数给构造函数,第一个参数就是使用了 Type Hinting,你可以称为类型约束或类型提示,来达到依赖注入。第二个参数就是简单的国家名称,默认 PRC,到这里我们国家有了,因为有名字了啊, 依赖一个团队 Team,但是 Team 还是空的,别急,让我们给 Team 加点球员,没有球员打 J8 球,还得是大牌,来定义一个 join 方法:

class Team
{
    protected $players = [];
    
    public function __construct($players = [])
    {
        $this->players = $players;
    }
    
    public function join(Player $player)
    {
        $this->players[] = $player;
    }
    
    public function getPlayers()
    {
        return $this->players;
    } 
}

首先,我们定义了一个 players 数组属性用来存储我们的球员,当我们组建一个团队的时候,我们将自动生成一个 players 球员数组来控制加入团队的球员。在 join 方法中再一次以 Type Hinting 的方式传递一个 Player 实例,只不过这一次实在普通方法里传递参数。最后用一个简单的获取方法 getPlayers 来获取我们加入到团队的球员,不然,这个 Team 类,是不是就只进不出了。接下来,我们要给我们的球员取个名字,要不然都不知道有哪些大牌加入我们的 Team 呢:

class Player 
{
    protected $name;
    
    public function __construct($name)
    {
        $this->name = $name;
    }
}

好像到目前为止,依赖关系比较清晰了呢,但我们还有个小问题,怎样让球员加入指定的国家呢?比如让梅西为我大天朝效力。所以,很显然国家必须有招募球员的能力:

发送消息

class Country
{
    protected $team;
    protected $name;
    
    public function __construct(Team $team, $name = 'PRC')
    {
        $this->team = $team;
        $this->name = $name;
    }
    
    public function recruit(Player $player)
    {
        $this->team->join($player);
    }
}

现在国家就用了招募球员到他们自己的团队里去了,这里我们可以将这个过程称为消息传递,message passing,方法的名称和输入的参数被概念化为消息,看成类与类,类与对象,对象与对象之间的通信。回头再梳理一遍依赖关系,正如前面所说,国家依赖团队,所以我们在 Country 的构造函数里传递了 Team 实例,那么实现这个过程,我们就应该首先创建一个 Team 实例,然后创建一个 Country 实例,并传递 Team 实例给 Country

$team = new Team;
 
$china = new Country($team);

OK,接下来招募 PlayersTeam 中:

$player1 = new Player('梅西');
$china->recruit($player1);

然后,让我们确认下是不是梅西就到了国家队了:

$team->getPlayers();


Array
(
    [0] => Player Object
        (
            [name:protected] => 梅西
        )
)

至此梅西已经来到国家队为我们征战世界杯了,诶,不对,足球不是一个人的游戏诶,梅西再牛逼没有队友怎么玩,继续招募球员:

$player2 = new Player('C罗');
$player3 = new Player('内马尔');
$player4 = new Player('科比');
$player5 = new Player('詹姆斯');
…………
…………
 
$usa->recruit($player2);
$usa->recruit($player3);
$usa->recruit($player4);
$usa->recruit($player5);


$team->getplayers();

至此,新一届的国家队已经成立,这阵容世界杯妥妥的。

总结:

很简单的依赖注入的小示例,首先我们并没有为了完成这个业务逻辑而创建一个大而全的类,试图将所有的功能放在一个位置,而是分成很多业务逻辑比较单一和清晰的小类,即使这些类是分开的, 他们也能够通过 message passing 或彼此调用方法来通信。是不是感受到松耦合,容易维护的地方了,比如足球队不要了,来个军队要开战了,招募点敢死队是不是很方便?

上一篇下一篇

猜你喜欢

热点阅读