听 Fabien Potencier 谈Symfony2 之 《What is Dependency Injection ?》

  什么是依赖注入?
  从PHP实现角度来分析依赖注入,因为PHP主要用于web开发,所以我们就看Web应用例子。
  为了克服HTTP协议的无状态性,web应用程序需要有一个途径来在web请求之间存储用户信息。最简单的方式是使用cookie或者采用更好一点的PHP内建的Session机制。

$_SESSION['language']='en';

  上面这句代码就实现了把语言存储到Session变量language里。从此之后,同一用户一些后续的请求都可以使用这个值了。因为$_SESSION 数组是个全局的变量。调用方式如下:

$user_language = $_SESSION['language'];

  因为依赖注入是面向对象世界里的概念,现在我们需要把PHP Session机制封装到一个类里面,然后应用到web应用程序中。

class SessionStorage

{

   function __construct($cookieName = 'PHP_SESS_ID')

   {

    session_name($cookieName);

    session_start();

   }

   

   function set($key, $value)

   {

    $_SESSION[$key] = $value;

   }



   function get($key)

   {

    return $_SESSION[$key];

   }



   //...

}

  接下来我们定义一个User类,来提供它的更高级的接口。

class User

{

    protected $storage;

    

    function __construct()

    {

    $this->$storage = new SessionStorage();

    }

    

    function setLanguage($language)

    {

    $this->storage->set('language', $language);

    }



    function getLanguage()

    {

    return $this->storage->get('language');

    }

   

    //.....

}

之后,我们就可以直接使用了。

$user = new User();

$user->setLanguage('en');

$user_language = $user->getLanguage();

  如果做得更加灵活一点呢?你想改变会话cookie的名字怎么办?有几个方式:可以在User类中在SessionStorage的构造函数中指定,即一种硬编码方式。

class User

{

   function __construct()

   {

      $this->storage = new SessionStorage('SESSION_ID');

   }

   //....

}

      或者在User类外面定义一个常量:STORAGE_SESSION_NAME 这种全局式的常量定义,不推荐。

class User

{

  function __construct()

  {

    $this->storage = new SessionStorage(STORAGE_SESSION_NAME);

  }

 

  // ...

}



define('STORAGE_SESSION_NAME','SESSION_ID');

  一种方式是把Session名作为User 类构造函数的参数:

class User

{

  function __construct($sessionName)

  {

    $this->storage = new SessionStorage($sessionName);

  }

 

  // ...

}

 

$user = new User('SESSION_ID');

    还有一种方式是PHP代码中常见的为Session存储类添加一个可选设置项数组options 

class User

{

  function __construct($storageOptions)

  {

    $this->storage = new SessionStorage($storageOptions['session_name']);

  }

 

  // ...

}

 

$user = new User(array('session_name' => 'SESSION_ID'));

  实现的方法有很多啊,无论是硬编码,设置为常量,作为构造函数的参数还是可选项数组都不是最佳选择。虽然后两者看起来好那么一点点,但它让User类的构造器堆积了一些跟自己没关系的参数。

  继续,另外一个问题又来了,如果我想改变SessionStorage类怎么办?比如在测试时,制造个假数据。或者你想把Session保存到数据库或者内存中时,就需要改变SessionStorage类了。就目前情况来看,如果不修改User类是不可能实现改变SessionStorage类的。

  现在我们考虑依赖注入,我们不在User类中创建SessionStorage对象,而是在外部创建后作为User类的构造方法参数传递给User对象。

class User

{

  function __construct($storage)

  {

    $this->storage = $storage;

  }

 

  // ...

}

  OK,这就是依赖注入,没别的了!现在再使用User类时可能需要比上次麻烦一点:

$storage = new SessionStorage('SESSION_ID');

$user = new User($storage);

  现在在不改变User类的前提下,改变Session名字,改变SessionStorage类,你想干啥都行了!

  总结一下:依赖注入是组件们通过他们的构造器,方法,或者属性字段来获取他们依赖的对象。

构造器注入:

class User

{

  function __construct($storage)

  {

    $this->storage = $storage;

  }

 

  // ...

}

设置器注入(方法注入):

class User

{

  function setSessionStorage($storage)

  {

    $this->storage = $storage;

  }

 

  // ...

}

属性字段注入:

class User

{

  public $sessionStorage;

}

 

$user->sessionStorage = $storage;

这些在Symfony 中你都会看到他们的身影:

构造注入:

$dispatcher = new sfEventDispatcher();

$storage = new sfMySQLSessionStorage(array('database' => 'session', 'db_table' => 'session'));

$user = new sfUser($dispatcher, $storage, array('default_culture' => 'en'));

将$dispatcher对象和$storage对象注入到$user对象中。

方法注入:

$transport = new Zend_Mail_Transport_Smtp('smtp.gmail.com', array(

  'auth'     => 'login',

  'username' => 'foo',

  'password' => 'bar',

  'ssl'      => 'ssl',

  'port'     => 465,

));

 

$mailer = new Zend_Mail();

$mailer->setDefaultTransport($transport);

该注入俗称为setter注入。

  Ok,这就是依赖注入了,它会在Symfony2中发展到服务容器注入。 以提供更加方便灵活松散耦合的多级依赖管理。它就是Service Container,它为Symfony2的执行效率和可扩展性提供了最强大的支持。

你可能感兴趣的:(dependency)