溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

怎么創(chuàng)建PHP DI容器

發(fā)布時間:2021-12-01 14:39:32 來源:億速云 閱讀:153 作者:iii 欄目:編程語言

這篇文章主要講解了“怎么創(chuàng)建PHP DI容器”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“怎么創(chuàng)建PHP DI容器”吧!

由開車開始

先開個車,為大家舉個栗子:

class Driver{    public function drive()
    {
        $car = new Car();        echo '老司機正在駕駛', $car->getCar(), PHP_EOL;
    }
}class Car{    protected $name = '普通汽車';    public function getCar()
    {        return $this->name;
    }
}

有兩個類,Driver和Car,老司機Driver有個方法driver,在調(diào)用的時候首先得整輛車$car,然后發(fā)車。大多數(shù)同學(xué)都寫過這樣或者類似的代碼,這樣的代碼單看沒啥毛病,挺正常的。但是,如果我要換輛車,開普通車撩不到妹。

class Benz extends Car{    protected $name = '奔馳';
}

這時候就需要做一個比較惡心的操作了,得改老司機的代碼了。(老司機:我做錯了什么?換輛車還得讓我重學(xué)駕照……)。因此我們需要把讓Car為外界注入,將Driver和Car解耦,不是老司機自己開車的時候還得自己去造車。于是就有了下面的結(jié)果

class Driver{    protected $car;    public function __construct(Car $car)
    {        $this->car = $car;
    }    public function drive()
    {        echo '老司機正在駕駛', $this->car->getCar(), PHP_EOL;
    }
}

此時Driver和Car兩個類已經(jīng)解耦,這兩個類的依賴,依靠上層代碼去管理。此時,老司機會這樣“開車”:

$car = new Car();
$driver = new Driver($car);
$driver->drive();

此時,我們創(chuàng)建Driver依賴的實例,并注入。上面的例子,我們實現(xiàn)了依賴注入,不過是手動的,寫起來感覺還是不爽。這么繁重的活怎么能手動來做呢,得讓程序自己去做。于是乎,DI容器誕生。

依賴注入容器

依賴注入與IoC模式類似工廠模式,是一種解決調(diào)用者和被調(diào)用者依賴耦合關(guān)系的模式。它解決了對象之間的依賴關(guān)系,使得對象只依賴IoC/DI容器,不再直接相互依賴,實現(xiàn)松耦合,然后在對象創(chuàng)建時,由IoC/DI容器將其依賴(Dependency)的對象注入(Inject)其內(nèi),這樣做可以最大程度實現(xiàn)松耦合。依賴注入說白一點,就是容器將某個類依賴的其他類的實例注入到這個類的實例中。

這段話可能說的有點抽象,回到剛才的例子吧。剛剛我手動完成了依賴注入,比較麻煩,如果一個大型的項目這樣做肯定會覺得很繁瑣,而且不夠優(yōu)雅。因此我們需要有一位總管代替我們?nèi)ジ蛇@個,這個總管就是容器。類的依賴管理全部交給容器去完成。因此,一般來說容器是一個全局的對象,大家共有的。

做一個自己的DI容器

寫一個功能,我們首先需要分析問題,因此我們先要明白,對于一個簡單的DI容器需要哪些功能,這直接關(guān)系到我們代碼的編寫。對于一個簡單的容器,至少需要滿足以下幾點:

  • 創(chuàng)建所需類的實例

  • 完成依賴管理(DI)

  • 可以獲取單例的實例

  • 全局唯一

綜上,我們的容器類大約長這樣:

class Container{    /**
     * 單例
     * @var Container
     */
    protected static $instance;    /**
     * 容器所管理的實例
     * @var array
     */
    protected $instances = [];    private function __construct(){}  
    private function __clone(){}    /**
     * 獲取單例的實例
     * @param string $class
     * @param array ...$params
     * @return object
     */
    public function singleton($class, ...$params)
    {}    /**
     * 獲取實例(每次都會創(chuàng)建一個新的)
     * @param string $class
     * @param array ...$params
     * @return object
     */
    public function get($class, ...$params)
    {}    /**
     * 工廠方法,創(chuàng)建實例,并完成依賴注入
     * @param string $class
     * @param array $params
     * @return object
     */
    protected function make($class, $params = [])
    {}    /**
     * @return Container
     */
    public static function getInstance()
    {        if (null === static::$instance) {            static::$instance = new static();
        }        return static::$instance;
    }
}

大體骨架已經(jīng)確定,接下來進入最核心的make方法:

protected function make($class, $params = []){  //如果不是反射類根據(jù)類名創(chuàng)建
  $class = is_string($class) ? new ReflectionClass($class) : $class;  //如果傳的入?yún)⒉粸榭眨瑒t根據(jù)入?yún)?chuàng)建實例
  if (!empty($params)) {    return $class->newInstanceArgs($params);
  }  //獲取構(gòu)造方法
  $constructor = $class->getConstructor();  //獲取構(gòu)造方法參數(shù)
  $parameterClasses = $constructor ? $constructor->getParameters() : [];  if (empty($parameterClasses)) {    //如果構(gòu)造方法沒有入?yún)?,直接?chuàng)建
    return $class->newInstance();
  } else {    //如果構(gòu)造方法有入?yún)?,迭代并遞歸創(chuàng)建依賴類實例
    foreach ($parameterClasses as $parameterClass) {
      $paramClass = $parameterClass->getClass();
      $params[] = $this->make($paramClass);
    }    //最后根據(jù)創(chuàng)建的參數(shù)創(chuàng)建實例,完成依賴的注入
    return $class->newInstanceArgs($params);
  }
}

為了容器的易用,我做了一些完善:

  • 實現(xiàn)ArrayAccess接口,使單例實例可以直接通過array的方式獲取,如果該實例沒有,則創(chuàng)建

  • 重寫__get方法,更方便的獲取

最終版:

class Container implements ArrayAccess{    /**
     * 單例
     * @var Container
     */
    protected static $instance;    /**
     * 容器所管理的實例
     * @var array
     */
    protected $instances = [];    private function __construct(){}    private function __clone(){}    /**
     * 獲取單例的實例
     * @param string $class
     * @param array  ...$params
     * @return object
     */
    public function singleton($class, ...$params)
    {        if (isset($this->instances[$class])) {            return $this->instances[$class];
        } else {            $this->instances[$class] = $this->make($class, $params);
        }        return $this->instances[$class];
    }    /**
     * 獲取實例(每次都會創(chuàng)建一個新的)
     * @param string $class
     * @param array  ...$params
     * @return object
     */
    public function get($class, ...$params)
    {        return $this->make($class, $params);
    }    /**
     * 工廠方法,創(chuàng)建實例,并完成依賴注入
     * @param string $class
     * @param array  $params
     * @return object
     */
    protected function make($class, $params = [])
    {        //如果不是反射類根據(jù)類名創(chuàng)建
        $class = is_string($class) ? new ReflectionClass($class) : $class;        //如果傳的入?yún)⒉粸榭?,則根據(jù)入?yún)?chuàng)建實例
        if (!empty($params)) {            return $class->newInstanceArgs($params);
        }        //獲取構(gòu)造方法
        $constructor = $class->getConstructor();        //獲取構(gòu)造方法參數(shù)
        $parameterClasses = $constructor ? $constructor->getParameters() : [];        if (empty($parameterClasses)) {            //如果構(gòu)造方法沒有入?yún)?,直接?chuàng)建
            return $class->newInstance();
        } else {            //如果構(gòu)造方法有入?yún)ⅲ⑦f歸創(chuàng)建依賴類實例
            foreach ($parameterClasses as $parameterClass) {
                $paramClass = $parameterClass->getClass();
                $params[] = $this->make($paramClass);
            }            //最后根據(jù)創(chuàng)建的參數(shù)創(chuàng)建實例,完成依賴的注入
            return $class->newInstanceArgs($params);
        }
    }    /**
     * @return Container
     */
    public static function getInstance()
    {        if (null === static::$instance) {            static::$instance = new static();
        }        return static::$instance;
    }    public function __get($class)
    {        if (!isset($this->instances[$class])) {            $this->instances[$class] = $this->make($class);
        }        return $this->instances[$class];
    }    public function offsetExists($offset)
    {        return isset($this->instances[$offset]);
    }    public function offsetGet($offset)
    {        if (!isset($this->instances[$offset])) {            $this->instances[$offset] = $this->make($offset);
        }        return $this->instances[$offset];
    }    public function offsetSet($offset, $value)
    {
    }    public function offsetUnset($offset) {        unset($this->instances[$offset]);
    }
}

現(xiàn)在借助容器我們寫一下上面的代碼:

$driver = $app->get(Driver::class);
$driver->drive();//output:老司機正在駕駛普通汽車復(fù)制代碼

就這么簡單,老司機就能發(fā)車。這里默認注入的是Car的實例,如果需要開奔馳,那只需要這樣:

$benz = $app->get(Benz::class);
$driver = $app->get(Driver::class, $benz);
$driver->drive();//output:老司機正在駕駛奔馳復(fù)制代碼

按照PSR-11的要求,依賴注入容器需要實現(xiàn)Psr\Container\ContainerInterface接口,這里只是演示并未去實現(xiàn),因為那需要引入Psr依賴庫,比較麻煩,其實也很簡單,只是多了幾個方法,有興趣的可以自己去了解下PSR-11的要求(傳送門)。

這里只是實現(xiàn)了一個非常簡陋的DI容器,實際中還需要考慮很多,而且這里的容器功能上還很簡陋。還有一些坑沒處理,比如出現(xiàn)循環(huán)依賴怎么處理、延遲加載的機制……

感謝各位的閱讀,以上就是“怎么創(chuàng)建PHP DI容器”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對怎么創(chuàng)建PHP DI容器這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI