本文實(shí)例講述了PHP基于反射機(jī)制實(shí)現(xiàn)自動(dòng)依賴(lài)注入的方法。分享給大家供大家參考,具體如下:
依賴(lài)注入又叫控制反轉(zhuǎn),使用過(guò)框架的人應(yīng)該都不陌生。很多人一看名字就覺(jué)得是非常高大上的東西,就對(duì)它望而卻步,今天抽空研究了下,解開(kāi)他它的神秘面紗。廢話(huà)不多說(shuō),直接上代碼;
/** * * 工具類(lèi),使用該類(lèi)來(lái)實(shí)現(xiàn)自動(dòng)依賴(lài)注入。 * */ class Ioc { // 獲得類(lèi)的對(duì)象實(shí)例 public static function getInstance($className) { $paramArr = self::getMethodParams($className); return (new ReflectionClass($className))->newInstanceArgs($paramArr); } /** * 執(zhí)行類(lèi)的方法 * @param [type] $className [類(lèi)名] * @param [type] $methodName [方法名稱(chēng)] * @param [type] $params [額外的參數(shù)] * @return [type] [description] */ public static function make($className, $methodName, $params = []) { // 獲取類(lèi)的實(shí)例 $instance = self::getInstance($className); // 獲取該方法所需要依賴(lài)注入的參數(shù) $paramArr = self::getMethodParams($className, $methodName); return $instance->{$methodName}(...array_merge($paramArr, $params)); } /** * 獲得類(lèi)的方法參數(shù),只獲得有類(lèi)型的參數(shù) * @param [type] $className [description] * @param [type] $methodsName [description] * @return [type] [description] */ protected static function getMethodParams($className, $methodsName = '__construct') { // 通過(guò)反射獲得該類(lèi) $class = new ReflectionClass($className); $paramArr = []; // 記錄參數(shù),和參數(shù)類(lèi)型 // 判斷該類(lèi)是否有構(gòu)造函數(shù) if ($class->hasMethod($methodsName)) { // 獲得構(gòu)造函數(shù) $construct = $class->getMethod($methodsName); // 判斷構(gòu)造函數(shù)是否有參數(shù) $params = $construct->getParameters(); if (count($params) > 0) { // 判斷參數(shù)類(lèi)型 foreach ($params as $key => $param) { if ($paramClass = $param->getClass()) { // 獲得參數(shù)類(lèi)型名稱(chēng) $paramClassName = $paramClass->getName(); // 獲得參數(shù)類(lèi)型 $args = self::getMethodParams($paramClassName); $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args); } } } } return $paramArr; } }
上面的代碼使用php的反射函數(shù),創(chuàng)建了一個(gè)容器類(lèi),使用該類(lèi)來(lái)實(shí)現(xiàn)其他類(lèi)的依賴(lài)注入功能。上面的依賴(lài)注入分為兩種,一種是構(gòu)造函數(shù)的依賴(lài)注入,一種是方法的依賴(lài)注入。 我們使用下面三個(gè)類(lèi)來(lái)做下測(cè)試。
class A { protected $cObj; /** * 用于測(cè)試多級(jí)依賴(lài)注入 B依賴(lài)A,A依賴(lài)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; /** * 測(cè)試構(gòu)造函數(shù)依賴(lài)注入 * @param A $a [使用引來(lái)注入A] */ public function __construct(A $a) { $this->aObj = $a; } /** * [測(cè)試方法調(diào)用依賴(lài)注入] * @param C $c [依賴(lài)注入C] * @param string $b [這個(gè)是自己手動(dòng)填寫(xiě)的參數(shù)] * @return [type] [description] */ public function bb(C $c, $b) { $c->cc(); echo "\r\n"; echo 'params:' . $b; } /** * 驗(yàn)證依賴(lài)注入是否成功 * @return [type] [description] */ public function bbb() { $this->aObj->aac(); } } class C { public function cc() { echo 'this is C->cc'; } }
測(cè)試構(gòu)造函數(shù)的依賴(lài)注入
// 使用Ioc來(lái)創(chuàng)建B類(lèi)的實(shí)例,B的構(gòu)造函數(shù)依賴(lài)A類(lèi),A的構(gòu)造函數(shù)依賴(lài)C類(lèi)。 $bObj = Ioc::getInstance('B'); $bObj->bbb(); // 輸出:this is C->cc , 說(shuō)明依賴(lài)注入成功。 // 打印$bObj var_dump($bObj); // 打印結(jié)果,可以看出B中有A實(shí)例,A中有C實(shí)例,說(shuō)明依賴(lài)注入成功。 object(B)#3 (1) { ["aObj":protected]=> object(A)#7 (1) { ["cObj":protected]=> object(C)#10 (0) { } } }
測(cè)試方法依賴(lài)注入
Ioc::make('B', 'bb', ['this is param b']); // 輸出結(jié)果,可以看出依賴(lài)注入成功。 this is C->cc params:this is param b
從上面兩個(gè)例子可以看出我們創(chuàng)建對(duì)象或者調(diào)用方法時(shí),根本就不用知道該類(lèi)或該方法依賴(lài)了那個(gè)類(lèi)。使用反射機(jī)制可以輕松的為我們自動(dòng)注入所需要的類(lèi)。
總結(jié)
好了,看到上面的代碼是不是覺(jué)得很簡(jiǎn)單,其實(shí)只要熟悉php的反射機(jī)制,依賴(lài)注入并不難實(shí)現(xiàn),上面的代碼為了方便理解,所以寫(xiě)的簡(jiǎn)單除暴,在實(shí)際的項(xiàng)目中肯定不會(huì)這么簡(jiǎn)單,比如:會(huì)對(duì)注入的類(lèi)和參數(shù)進(jìn)行配置,比如會(huì)緩存實(shí)例化過(guò)的類(lèi),下次需要該類(lèi)的實(shí)例時(shí),可以直接使用,而不用在重新初始化,等等。不過(guò)相信原理了解了,其他的可以隨著項(xiàng)目的需求自己去完善。