美文网首页
初识依赖注入

初识依赖注入

作者: kylesean | 来源:发表于2017-01-09 23:07 被阅读36次

    依赖注入,又称为控制反转(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 或彼此调用方法来通信。是不是感受到松耦合,容易维护的地方了,比如足球队不要了,来个军队要开战了,招募点敢死队是不是很方便?

    相关文章

      网友评论

          本文标题:初识依赖注入

          本文链接:https://www.haomeiwen.com/subject/mmzhbttx.html