听 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的执行效率和可扩展性提供了最强大的支持。

posted @ 2012-06-20 16:43  Seekr  阅读(2283)  评论(2编辑  收藏  举报