依賴注入又叫控制反轉(zhuǎn),使用過框架的人應(yīng)該都不陌生
2021-08-19
依賴注入也稱為控制反轉(zhuǎn)。任何使用過該框架的人都應(yīng)該熟悉它。
很多人一看名字就覺得是個很高大上的東西,望而卻步php網(wǎng)站注入工具,今天揭開了它的神秘面紗。
廢話不多說,直接上代碼;
/** * * 工具類,使用該類來實現(xiàn)自動依賴注入。 * */ class Ioc { // 獲得類的對象實例 public static function getInstance($className) { $paramArr = self::getMethodParams($className); return (new ReflectionClass($className))->newInstanceArgs($paramArr); } /** * 執(zhí)行類的方法 * @param [type] $className [類名] * @param [type] $methodName [方法名稱] * @param [type] $params [額外的參數(shù)] * @return [type] [description] */ public static function make($className, $methodName, $params = []) { // 獲取類的實例 $instance = self::getInstance($className); // 獲取該方法所需要依賴注入的參數(shù) $paramArr = self::getMethodParams($className, $methodName); return $instance->{$methodName}(...array_merge($paramArr, $params)); } /** * 獲得類的方法參數(shù),只獲得有類型的參數(shù) * @param [type] $className [description] * @param [type] $methodsName [description] * @return [type] [description] */ protected static function getMethodParams($className, $methodsName = '__construct') { // 通過反射獲得該類 $class = new ReflectionClass($className); $paramArr = []; // 記錄參數(shù),和參數(shù)類型 // 判斷該類是否有構(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ù)類型 foreach ($params as $key => $param) { if ($paramClass = $param->getClass()) { // 獲得參數(shù)類型名稱 $paramClassName = $paramClass->getName(); // 獲得參數(shù)類型 $args = self::getMethodParams($paramClassName); $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args); } } } } return $paramArr; } }
以上代碼利用PHP的反射函數(shù)創(chuàng)建了一個容器類,用于實現(xiàn)其他類的依賴注入功能。
上面的依賴注入分為兩種,一種是構(gòu)造函數(shù)的依賴注入,一種是方法的依賴注入。
我們使用以下三個類進(jìn)行測試。
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; /** * 測試構(gòu)造函數(shù)依賴注入 * @param A $a [使用引來注入A] */ public function __construct(A $a) { $this->aObj = $a; } /** * [測試方法調(diào)用依賴注入] * @param C $c [依賴注入C] * @param string $b [這個是自己手動填寫的參數(shù)] * @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'; } }
測試構(gòu)造函數(shù)的依賴注入:
// 使用Ioc來創(chuàng)建B類的實例,B的構(gòu)造函數(shù)依賴A類,A的構(gòu)造函數(shù)依賴C類。 $bObj = Ioc::getInstance('B'); $bObj->bbb(); // 輸出:this is C->cc , 說明依賴注入成功。 // 打印$bObj var_dump($bObj); // 打印結(jié)果,可以看出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']); // 輸出結(jié)果,可以看出依賴注入成功。 this is C->cc params:this is param b
從上面兩個例子可以看出,當(dāng)我們創(chuàng)建一個對象或者調(diào)用一個方法時,我們不需要知道依賴哪個類或者哪個方法。
使用反射機(jī)制可以輕松地自動注入我們需要的類。
總結(jié):
其實只要熟悉PHP的反射機(jī)制,依賴注入并不難實現(xiàn)。上面的代碼很容易理解,所以寫起來很簡單,消除暴力。實際項目中肯定不會這么簡單。
比如:會配置注入的類和參數(shù)php網(wǎng)站注入工具,比如實例化的類會被緩存,下次需要類的實例時,可以直接使用,無需重新初始化等
只要理解了原理,就可以根據(jù)項目的需要改進(jìn)其他的。