PHP反射機制實現自動依賴注入

依賴注入又叫控制反轉,使用過框架的人應該都不陌生。很多人一看名字就覺得是非常高大上的東西,就對它望而卻步,今天抽空研究了下,解開他它的神秘面紗。廢話不多說,直接上代碼;

/**
*
* 工具類,使用該類來實現自動依賴注入。
*
*/
class Ioc {

    // 獲得類的對象實例
    public static function getInstance($className) {

        $paramArr = self::getMethodParams($className);

        return (new ReflectionClass($className))->newInstanceArgs($paramArr);
    }

    /**
     * 執行類的方法
     * @param  [type] $className  [類名]
     * @param  [type] $methodName [方法名稱]
     * @param  [type] $params     [額外的參數]
     * @return [type]             [description]
     */
    public static function make($className, $methodName, $params = []) {

        // 獲取類的實例
        $instance = self::getInstance($className);

        // 獲取該方法所需要依賴注入的參數
        $paramArr = self::getMethodParams($className, $methodName);

        return $instance->{$methodName}(...array_merge($paramArr, $params));
    }

    /**
     * 獲得類的方法參數,只獲得有類型的參數
     * @param  [type] $className   [description]
     * @param  [type] $methodsName [description]
     * @return [type]              [description]
     */
    protected static function getMethodParams($className, $methodsName = '__construct') {

        // 通過反射獲得該類
        $class = new ReflectionClass($className);
        $paramArr = []; // 記錄參數,和參數類型

        // 判斷該類是否有構造函數
        if ($class->hasMethod($methodsName)) {
            // 獲得構造函數
            $construct = $class->getMethod($methodsName);

            // 判斷構造函數是否有參數
            $params = $construct->getParameters();

            if (count($params) > 0) {

                // 判斷參數類型
                foreach ($params as $key => $param) {

                    if ($paramClass = $param->getClass()) {

                        // 獲得參數類型名稱
                        $paramClassName = $paramClass->getName();

                        // 獲得參數類型
                        $args = self::getMethodParams($paramClassName);
                        $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args);
                    }
                }
            }
        }

        return $paramArr;
    }
}

上面的代碼使用php的反射函數,創建了一個容器類,使用該類來實現其他類的依賴注入功能。上面的依賴注入分為兩種,一種是構造函數的依賴注入,一種是方法的依賴注入。 我們使用下面三個類來做下測試。

class A {

    protected $cObj;

    /**
     * 用于測試多級依賴注入 B依賴A,A依賴C
     * @param C $c [description]
     */
    public function __construct(C $c) {

        $this->cObj = $c;
    }

    public function aa() {

        echo 'this is A->test';
    }

    public function aac() {

        $this->cObj->cc();
    }
}

class B {

    protected $aObj;

    /**
     * 測試構造函數依賴注入
     * @param A $a [使用引來注入A]
     */
    public function __construct(A $a) {

        $this->aObj = $a;
    }

    /**
     * [測試方法調用依賴注入]
     * @param  C      $c [依賴注入C]
     * @param  string $b [這個是自己手動填寫的參數]
     * @return [type]    [description]
     */
    public function bb(C $c, $b) {

        $c->cc();
        echo "\r\n";

        echo 'params:' . $b;
    }

    /**
     * 驗證依賴注入是否成功
     * @return [type] [description]
     */
    public function bbb() {

        $this->aObj->aac();
    }
}

class C {

    public function cc() {

        echo 'this is C->cc';
    }
}

測試構造函數的依賴注入

// 使用Ioc來創建B類的實例,B的構造函數依賴A類,A的構造函數依賴C類。
$bObj = Ioc::getInstance('B');
$bObj->bbb(); // 輸出:this is C->cc , 說明依賴注入成功。

// 打印$bObj
var_dump($bObj);

// 打印結果,可以看出B中有A實例,A中有C實例,說明依賴注入成功。
object(B)#3 (1) {
  ["aObj":protected]=>
  object(A)#7 (1) {
    ["cObj":protected]=>
    object(C)#10 (0) {
    }
  }
}

測試方法依賴注入

Ioc::make('B', 'bb', ['this is param b']);

// 輸出結果,可以看出依賴注入成功。
this is C->cc
params:this is param b

從上面兩個例子可以看出我們創建對象或者調用方法時,根本就不用知道該類或該方法依賴了那個類。使用反射機制可以輕松的為我們自動注入所需要的類。

總結
好了,看到上面的代碼是不是覺得很簡單,其實只要熟悉php的反射機制,依賴注入并不難實現,上面的代碼為了方便理解,所以寫的簡單除暴,在實際的項目中肯定不會這么簡單,比如:會對注入的類和參數進行配置,比如會緩存實例化過的類,下次需要該類的實例時,可以直接使用,而不用在重新初始化,等等。不過相信原理了解了,其他的可以隨著項目的需求自己去完善。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 轉自 http://blog.csdn.net/qq_20678155/article/details/70158...
    E狼閱讀 1,241評論 0 7
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,869評論 18 139
  • 1、通過CocoaPods安裝項目名稱項目信息 AFNetworking網絡請求組件 FMDB本地數據庫組件 SD...
    陽明AGI閱讀 16,003評論 3 119
  • 我們單位愛扣錢,這個是單位的一大特色。 高貴氣質的差(chai)人總是能找出千兒八百條規章制度約束你,像念說明書一...
    小魚叔閱讀 591評論 2 1
  • 外觀模式:為系統中的一組接口提供一個統一的接口。外觀定義一個高層的接口,讓子系統更易于使用。 外觀的使用場景 子系...
    863cda997e42閱讀 61評論 0 0