php依赖注入 简书,php设计模式 依赖注入和控制反转

依赖注入和控制反转在现如今的主流框架中经常可以看到,主要是用于解决程序的松耦合,便于程序的协调开发和管理。

DI——Dependency Injection 依赖注入

IoC——Inversion of Control 控制反转

当一个类的实例需要另一个类的实例协助时,在传统的程序设计过程中,通常由调用者来创建被调用者的实例。而采用依赖注入的方式,创建被调用者的工作不再由调用者来完成,因此叫控制反转,创建被调用者的实例的工作由IOC容器来完成,然后注入调用者,因此也称为依赖注入。

控制反转(Inversion of Control):当调用者需要被调用者的协助时,在传统的程序设计过程中,通常由调用者来创建被调用者的实例,但在这里,创建被调用者的工作不再由调用者来完成,而是将被调用者的创建移到调用者的外部,从而反转被调用者的创建,消除了调用者对被调用者创建的控制,因此称为控制反转。

依赖注入(Dependency Injection):要实现控制反转,解决方案是将创建被调用者实例的工作交由IoC容器来完成,然后在调用者中注入被调用者(通过构造器/方法注入实现),这样我们就实现了调用者与被调用者的解耦,该过程被称为依赖注入。依赖注入是控制反转的一种实现方式。常见注入方式有三种:setter、constructor injection、property injection。

比如我们写一个file的缓存类,类user需要用到这个缓存类 传统的调用方式是在调用者中实例化被调用者

class fileCache{

/**

* 设置缓存

* @param $name

* @param $value

*/

public function set($name ,$value){

//TODO

}

/**

* 删除缓存

* @param $name

*/

public function delete($name){

//TODO

}

/**

* 获取缓存数据

* @param $name

*/

public function get($name){

//TODO

}

}

class User {

protected $cache;

public function __construct() {

$this->cache = new fileCache();

}

public function getUser(){

return $this->cache->get("user");

}

}

$user = new User();

$user->getUser();

一旦平台或者环境发生变化,我们不用file来做为缓存,而是用redis做缓存,那相应调用类User就需要做出修改。程序不应该依赖于具体的实现,而是要依赖抽像的接口

class fileCache implements Cache{

public function set($name)

{

//TODO

}

public function get($name)

{

//TODO

}

public function delete($name)

{

//TODO

}

}

class redisCache implements Cache{

public function set($name)

{

//TODO

}

public function get($name)

{

//TODO

}

public function delete($name)

{

//TODO

}

}

构造器注入

class User{

protected $cache ;

public function __construct(Cache $cache)

{

$this->cache = $cache;

}

public function getUser()

{

return $this->cache->get('user');

}

}

$redis = new redisCache();

$user = new User($redis);

$user->getUser();

$file = new fileCache();

$user = new User($file);

$user->getUser();

setter注入

class User{

protected $cache ;

public function setCache( Cache $cache)

{

$this->cache = $cache;

}

public function getUser()

{

return $this->cache->get('user');

}

}

$redis = new redisCache();

$user = new User();

$user->setCache($redis);

$user->getUser();

$file = new fileCache();

$user = new User();

$user->setCache($file);

$user->getUser();

但是这样的注入还会产生一个问题,当调用类依赖于多个外部类时,我们就需要不断的写set

$user = new User();

$user->setDb($db);//注入db连接

$user->setCache($file);//注入缓存处理类

#.....

如果引入的外部类很多时,就会变得十分繁琐。这就引入另一个概念容器又叫做IoC容器、DI容器。

这里我们引入一个约定:在User类的构造函数里传入一个名为Di $di的参数,如下:

class Di{

protected $_objects = [];

public function set($name, $object)

{

$this->_objects[$name] = $object;

}

public function get($name) {

return $this->_objects[$name];

}

}

class User {

private $_di;

function __construct(Di &$di)

{

$this->_di = $di;

}

//通过di容器获取db实例

public function getUser()

{

return $this->_di->get('cache')->get('User');

}

}

$di = new Di();

$cache = new fileCache();

$di->set("cache",$cache);

$user = new User($di);

$user->getUser();

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值