PHP经验分享PHP开发PHP成长之路

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

2020-05-25  本文已影响0人  juggleczg

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

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();
上一篇 下一篇

猜你喜欢

热点阅读