首页 > 代码库 > 再说php依赖注入

再说php依赖注入

前段时间,有朋友问我yii2的依赖注入是怎么个玩法,好吧,

经常看到却一直不甚理解的概念,这里我再对自己认识的依赖注入深刻的表达下我的理解,依赖注入(DI)以及控制器反转(Ioc)。 依赖注入就是组件通过构造器,方法或者属性字段来获取相应的依赖对象。

举个现实生活中的例子来理解, 比如我要一把菜刀 如何获得
1.可以自己造一把,对应new一个。
2.可以找生产菜刀的工厂去买一把,对应工厂模式。
3.可以打电话 让店家送货上门,对应依赖注入

依赖注入(DI)的概念虽然听起来很深奥,但是如果你用过一些新兴的php框架的话,对于DI一定不陌生,因为它们多多少少都用到了依赖注入来处理类与类之间的依赖关系。

php中传递依赖关系的三种方案

其实要理解DI,首先要明白在php中如何传递依赖关系。 
第一种方案,也是最不可取的方案,就是在A类中直接用new关键词来创建一个B类,如下代码所示:

<?php
class A
{
    public function __construct()
    {
        $b = new B();
    }

}

为什么这种方案不可取呢?因为这样的话,A与B就耦合在了一起,也就是说A类无法脱离B类工作。 
第二种方案就是在A类的方法中传入需要的B类,如下代码所示:

<?php
class A
{
    public function __construct(B $b)
    {

    }

}

这种方法比第一种方案有了改进,A类不必与B类捆绑在一起,只要传入的类满足A类的需求,也可以是C类,也可以是D类等等。 
但是这种方案的弊端在于如果A类依赖的类较多,参数列表会很长,容易发生混乱。 
第三种方案是使用set方法传入,如下代码所示:

<?php
class A
{
    public function setB(B $b)
    {
        $this->b = $b;
    }

}

这种方案同样存在和第二种方案一样的弊端,当依赖的类增多时,我们需要些很多很多的set方法。

这时我们在想如果有一个专门的类(或者说一个容器)可以帮我们管理这些依赖关系就好了。

一个简单的依赖注入的例子

如下代码来自twittee:

<?php
class Container {
 private $s=array();
 function __set($k, $c) { $this->s[$k]=$c; }
 function __get($k) { return $this->s[$k]($this); }
}

有了container类之后我们可以怎样管理A与B之间的依赖关系呢,用代码说话吧:

<?php
class A
{
    private $container;

    public function __construct(Container $container)
    {
        $this->container = $container;
    }

    public function doSomeThing()
    {
        //do something which needs class B
        $b = $this->container->getB();

        //to do 
    }

}

再将B类注入到容器类中:

$c = new Container();
$c->setB(new B());

还可以传入一个匿名函数,这样B类就不会在传入时就立即实例化,而是在真正调用时才完成实例化的工作:

$c = new Container();
$c->setB(function (){
    return new B();
});

这里举的只是一个很简单的例子,在实际中,容器类要考虑的有很多,比如延迟加载等等。

再比如我是一个演员,我不可能要求某个导演,我要演某某剧的男一号,相反,导演可以决定让谁来演。而我们的object就是这个演员。

注入的几个途径:
1.construct注入

<?php
class Book {
   private $db_conn;
 
   public function __construct($db_conn) {
       $this->db_conn = $db_conn;
   }
}

但是如果依赖过多,那么在构造方法里必然传入多个参数,三个以上就会使代码变的难以阅读。

2.set注入

<?php
  $book = new Book();
  $book->setdb($db);
  $book->setprice($price);
  $book->set_author($author);
?>

代码很清晰,但是当我们需要注入第四个依赖时,意味着又要增加一行。

比较好的解决办法是 建立一个class作为所有依赖关系的container,在这个class中可以存放、创建、获取、查找需要的依赖关系

<?php
class Ioc {
   protected $db_conn;
   public static function make_book() {
       $new_book = new Book();
       $new_book->set_db(self::$db_conn);
       //...
       //...
       //其他的依赖注入
       return $new_book;
   }
}

此时,如果获取一个book实例,只需要执行$newone = Ioc::makebook();

以上是container的一个具体实例,最好还是不要把具体的某个依赖注入写成方法,采用registry注册,get获取比较好。

<?php
class Ioc {
/**
* @var 注册的依赖数组
*/
 
   protected static $registry = array();
 
   /**
    * 添加一个resolve到registry数组中
    * @param  string $name 依赖标识
    * @param  object $resolve 一个匿名函数用来创建实例
    * @return void
    */
   public static function register($name, Closure $resolve)
   {
      static::$registry[$name] = $resolve;
   }
 
   /**
     * 返回一个实例
     * @param  string $name 依赖的标识
     * @return mixed
     */
   public static function resolve($name)
   {
       if ( static::registered($name) )
       {
          $name = static::$registry[$name];
          return $name();
       }
       throw new Exception(‘Nothing registered with that name, fool.‘);
   }
   /**
    * 查询某个依赖实例是否存在
    * @param  string $name id
    * @return bool 
    */
   public static function registered($name)
   {
      return array_key_exists($name, static::$registry);
   }
}

现在就可以通过如下方式来注册和注入一个依赖

<?php
$book = Ioc::registry(‘book‘, function(){
$book = new Book;
$book->setdb(‘...‘);
$book->setprice(‘...‘);
return $book;
});
 
//注入依赖
$book = Ioc::resolve(‘book‘);
?>

再说php依赖注入