在laravel中,依賴注入的實現原理是利用類方法反射,取得參數類型,然后利用容器構造好實例,再使用回調函數調起;注入對象構造函數不能有參數,否則會報錯,依賴注入必須要由Router類調起,否則直接用new方式是無法實現注入的。
本文操作環境:Windows10系統、Laravel6版、Dell G3電腦。
Laravel的依賴注入實現原理是什么
laravel容器包含控制反轉和依賴注入,使用起來就是,先把對象bind好,需要時可以直接使用make來取就好。
具體分析參照:http://laravelacademy.org/post/769.html
通常我們的調用如下。
$config?=?$container->make('config'); $connection?=?new?Connection($this->config);
比較好理解,這樣的好處就是不用直接 new 一個實例了,方法傳值沒啥改變,還可以多處共享此實例。
但這跟依賴注入有什么關系,真正的依賴注入是不需給方法傳遞任何參數值,只需要指明方法參數類型,代碼自動查找關系依賴自動注入。
這個特性在 laravel 的 Controller、Job 等處可以體現,如下:
class?TestController?extends?Controller { ????public?function?anyConsole(Request?$request,?Auth?$input) ????{ ????????//todo ????} }
我們來看下他是怎么實現自動依賴注入的:
由 index.PHP 調用 Kernel ,經過多層 Kernel 管道調用,再到 Router ,經過多層中間件管道調用。最終定位到
Illuminate/Routing/Route.php 第124行。
public?function?run(Request?$request) { ????$this->container?=?$this->container??:?new?Container; ????try?{ ????????if?(!?is_string($this->action['uses']))?{ ????????????return?$this->runCallable($request); ????????} ????????if?($this->customDispatcherIsBound())?{ ????????????return?$this->runWithCustomDispatcher($request); ????????} ????????return?$this->runController($request); ????}?catch?(HttpResponseException?$e)?{ ????????return?$e->getResponse(); ????} }
判斷 $this->action[‘uses’](格式行如:AppHttpControllerDatacenterRealTimeController@anyConsole)是否字符串, $this->customDispatcherIsBound判斷是否綁定了用戶自定義路由。然后跳轉到 $this->runController($request)。
protected?function?runController(Request?$request) { ????list($class,?$method)?=?explode('@',?$this->action['uses']); ????$parameters?=?$this->resolveClassMethodDependencies( ????????$this->parametersWithoutNulls(),?$class,?$method ????); ????if?(!?method_exists($instance?=?$this->container->make($class),?$method))?{ ????????throw?new?NotFoundHttpException; ????} ????return?call_user_func_array([$instance,?$method],?$parameters); }
$this->resolveClassMethodDependencies 這個方法一看名字就知道是我們要找的方法。$this->parametersWithoutNulls()是過濾空字符,$class、$method分別行如:AppHttpControllerDatacenterRealTimeController 與 anyConsole。
protected?function?resolveClassMethodDependencies(array?$parameters,?$instance,?$method) { ????if?(!?method_exists($instance,?$method))?{ ????????return?$parameters; ????} ????return?$this->resolveMethodDependencies( ????????$parameters,?new?ReflectionMethod($instance,?$method) ????); }
new ReflectionMethod($instance, $method) 是拿到類方法的反射對象,參見文檔:http://www.php.net/manual/zh/class.reflectionmethod.php
下面跳轉到Illuminate/Routing/RouteDependencyResolverTrait.php 第54行。
public?function?resolveMethodDependencies(array?$parameters,?ReflectionFunctionAbstract?$reflector) { ????$originalParameters?=?$parameters; ????foreach?($reflector->getParameters()?as?$key?=>?$parameter)?{ ????????$instance?=?$this->transformDependency( ????????????$parameter,?$parameters,?$originalParameters ????????); ????????if?(!?is_null($instance))?{ ????????????$this->spliceIntoParameters($parameters,?$key,?$instance); ????????} ????} ????return?$parameters; }
通過反射類方法得到類參數數組,然后遍歷傳遞給 $this->transformDependency 方法。如果實例獲取不到則調用 $this->spliceIntoParameters 清楚該參數。
protected?function?transformDependency(ReflectionParameter?$parameter,?$parameters,?$originalParameters) { ????$class?=?$parameter->getClass(); ????if?($class?&&?!?$this->alreadyInParameters($class->name,?$parameters))?{ ????????return?$this->container->make($class->name); ????} }
終于看到了容器的影子,沒錯最終對象還是通過容器的 make 方法取出來的。至此參數就構造好了,然后最終會被 runController 方法的 call_user_func_array 回調。
總結:
依賴注入原理其實就是利用類方法反射,取得參數類型,然后利用容器構造好實例。然后再使用回調函數調起。
注入對象構造函數不能有參數。否則會報錯。Missing argument 1
依賴注入故然好,但它必須要由 Router 類調起,否則直接用 new方式是無法實現注入的。所以這就為什么只有 Controller 、Job 類才能用這個特性了。
【相關推薦:laravel視頻教程】