首頁 >php框架 >Laravel >Laravel的依賴注入實作原理是什麼

Laravel的依賴注入實作原理是什麼

WBOY
WBOY原創
2022-02-18 11:06:446182瀏覽

在Laravel中,依賴注入的實作原理是利用類別方法反射,取得參數類型,然後利用容器建構好實例,再使用回呼函數調起;注入物件建構子不能有參數,否則會報錯,依賴注入必須要由Router類別調起,否則直接用new方式是無法實現注入的。

Laravel的依賴注入實作原理是什麼

本文操作環境: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'](格式行如:\App\Http\Controller\Datacenter\RealTimeController@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分別行如:\App\Http\Controller\Datacenter\RealTimeController 與 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影片教學

以上是Laravel的依賴注入實作原理是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn