這篇文章介紹的內容是關於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 , 說明依賴注入成功。// 列印$bObjvar_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->ccparams:this is param b
從上面兩個例子可以看出我們建立對象或者調用方法時,根本就不用知道該類或該方法依賴了那個類。使用反射機制可以輕鬆的為我們自動注入所需要的類。
總結
好了,看到上面的代碼是不是覺得很簡單,其實只要熟悉php的反射機制,依賴注入並不難實現,上面的代碼為了方便理解,所以寫的簡單除暴,在實際的項目中肯定不會這麼簡單,比如:會對注入的類和參數進行配置,比如會緩衝執行個體化過的類,下次需要該類的執行個體時,可以直接使用,而不用在重新初始化,等等。不過相信原理瞭解了,其他的可以隨著項目的需求自己去完善。