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