ThinkPHP6源碼:從Http類的實例化看依賴註入是如何實現的

来源:https://www.cnblogs.com/a609251438/archive/2019/12/27/12109648.html
-Advertisement-
Play Games

ThinkPHP 6 從原先的 App 類中分離出 Http 類,負責應用的初始化和調度等功能,而 App 類則專註於容器的管理,符合單一職責原則。 以下源碼分析,我們可以從 App,Http 類的實例化過程,瞭解類是如何實現自動實例化的,依賴註入是怎麼實現的。 從入口文件出發 當訪問一個 Thin ...


ThinkPHP 6 從原先的 App 類中分離出 Http 類,負責應用的初始化和調度等功能,而 App 類則專註於容器的管理,符合單一職責原則。

以下源碼分析,我們可以從 AppHttp 類的實例化過程,瞭解類是如何實現自動實例化的,依賴註入是怎麼實現的。

 

從入口文件出發

當訪問一個 ThinkPHP 搭建的站點,框架最先是從入口文件開始的,然後才是應用初始化、路由解析、控制器調用和響應輸出等操作。

入口文件主要代碼如下:

// 引入自動載入器,實現類的自動載入功能(PSR4標準)

// 對比Laravel、Yii2、Thinkphp的自動載入實現,它們基本就都一樣

// 具體實現可參考我之前寫的Laravel的自動載入實現:

// @link: https://learnku.com/articles/20816

require __DIR__ . '/../vendor/autoload.php';

// 這一句和分為兩部分分析,App的實例化和調用「http」,具體見下文分析

$http = (new App())->http;

$response = $http->run();

$response->send();

$http->end($response);

 

App 實例化

執行 new App() 實例化時,首先會調用它的構造函數。

public function __construct(string $rootPath = '')

{

    // thinkPath目錄:如,D:\dev\tp6\vendor\topthink\framework\src\

    $this->thinkPath   = dirname(__DIR__) . DIRECTORY_SEPARATOR;

    // 項目根目錄,如:D:\dev\tp6\

    $this->rootPath    = $rootPath ? rtrim($rootPath, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR : $this->getDefaultRootPath();

    $this->appPath     = $this->rootPath . 'app' . DIRECTORY_SEPARATOR;

    $this->runtimePath = $this->rootPath . 'runtime' . DIRECTORY_SEPARATOR;

    // 如果存在「綁定類庫到容器」文件

    if (is_file($this->appPath . 'provider.php')) {

        //將文件里的所有映射合併到容器的「$bind」成員變數中

        $this->bind(include $this->appPath . 'provider.php');

    }

    //將當前容器實例保存到成員變數「$instance」中,也就是容器自己保存自己的一個實例

    static::setInstance($this);

    // 保存綁定的實例到「$instances」數組中,見對應分析

    $this->instance('app', $this);

    $this->instance('think\Container', $this);

}

 

構造函數實現了項目各種基礎路徑的初始化,並讀取了 provider.php 文件,將其類的綁定併入 $bind 成員變數,provider.php 文件預設內容如下:

return [

    'think\Request'          => Request::class,

    'think\exception\Handle' => ExceptionHandle::class,

];

 

合併後,$bind 成員變數的值如下:

$bind 的值是一組類的標識到類的映射。從這個實現也可以看出,我們不僅可以在 provider.php 文件中添加標識到類的映射,而且可以覆蓋其原有的映射,也就是將某些核心類替換成自己定義的類。

static::setInstance($this) 實現的作用,如圖:

think\App 類的 $instance 成員變數指向 think\App 類的一個實例,也就是類自己保存自己的一個實例。

instance() 方法的實現:

public function instance(string $abstract, $instance)

{

    //檢查「$bind」中是否保存了名稱到實際類的映射,如 'app'=> 'think\App'

    //也就是說,只要綁定了這種對應關係,通過傳入名稱,就可以找到實際的類

    if (isset($this->bind[$abstract])) {

        //$abstract = 'app', $bind = "think\App"

        $bind = $this->bind[$abstract];

        //如果「$bind」是字元串,重走上面的流程

        if (is_string($bind)) {

            return $this->instance($bind, $instance);

        }

    }

    //保存綁定的實例到「$instances」數組中

    //比如,$this->instances["think\App"] = $instance;

    $this->instances[$abstract] = $instance;

    return $this;

}

 

執行結果大概是這樣的:

Http 類的實例化以及依賴註入原理

這裡,$http = (new App())->http,前半部分好理解,後半部分乍一看有點讓人摸不著頭腦,App 類並不存在 http 成員變數,這裡何以大膽調用了一個不存在的東東呢?

原來,App 類繼承自 Container 類,而 Container 類實現了__get() 魔術方法,在 PHP 中,當訪問到的變數不存在,就會觸發__get() 魔術方法。該方法的實現如下:

public function __get($name)

{

    return $this->get($name);

}

 

實際上是調用 get() 方法:

public function get($abstract)

{

    //先檢查是否有綁定實際的類或者是否實例已存在

    //比如,$abstract = 'http'

    if ($this->has($abstract)) {

        return $this->make($abstract);

    }

    // 找不到類則拋出類找不到的錯誤

    throw new ClassNotFoundException('class not exists: ' . $abstract, $abstract);

}

 

然而,實際上,主要是 make() 方法:

public function make(string $abstract, array $vars = [], bool $newInstance = false)

    {

        //如果已經存在實例,且不強制創建新的實例,直接返回已存在的實例

        if (isset($this->instances[$abstract]) && !$newInstance) {

            return $this->instances[$abstract];

        }

        //如果有綁定,比如 'http'=> 'think\Http',則 $concrete = 'think\Http'

        if (isset($this->bind[$abstract])) {

            $concrete = $this->bind[$abstract];

            if ($concrete instanceof Closure) {

                $object = $this->invokeFunction($concrete, $vars);

            } else {

                //重走一遍make函數,比如上面http的例子,則會調到後面「invokeClass()」處

                return $this->make($concrete, $vars, $newInstance);

            }

        } else {

            //實例化需要的類,比如'think\Http'

            $object = $this->invokeClass($abstract, $vars);

        }

        if (!$newInstance) {

            $this->instances[$abstract] = $object;

        }

        return $object;

    }

 

然而,然而,make() 方法主要靠 invokeClass() 來實現類的實例化。該方法具體分析:

public function invokeClass(string $class, array $vars = [])

    {

        try {

            //通過反射實例化類

            $reflect = new ReflectionClass($class);

            //檢查是否有「__make」方法

            if ($reflect->hasMethod('__make')) {

                //返回的$method包含'__make'的各種信息,如公有/私有

                $method = new ReflectionMethod($class, '__make');

                //檢查是否是公有方法且是靜態方法

                if ($method->isPublic() && $method->isStatic()) {

                    //綁定參數

                    $args = $this->bindParams($method, $vars);

                    //調用該方法(__make),因為是靜態的,所以第一個參數是null

                    //因此,可得知,一個類中,如果有__make方法,在類實例化之前會首先被調用

                    return $method->invokeArgs(null, $args);

                }

            }

            //獲取類的構造函數

            $constructor = $reflect->getConstructor();

            //有構造函數則綁定其參數

            $args = $constructor ? $this->bindParams($constructor, $vars) : [];

            //根據傳入的參數,通過反射,實例化類

            $object = $reflect->newInstanceArgs($args);

            // 執行容器回調

            $this->invokeAfter($class, $object);

            return $object;

        } catch (ReflectionException $e) {

            throw new ClassNotFoundException('class not exists: ' . $class, $class, $e);

        }

    }

 

以上代碼可看出,在一個類中,添加__make() 方法,在類實例化時,會最先被調用。以上最值得一提的是 bindParams() 方法:

protected function bindParams($reflect, array $vars = []): array

{

    //如果參數個數為0,直接返回

    if ($reflect->getNumberOfParameters() == 0) {

        return [];

    }

    // 判斷數組類型 數字數組時按順序綁定參數

    reset($vars);

    $type   = key($vars) === 0 ? 1 : 0;

    //通過反射獲取函數的參數,比如,獲取Http類構造函數的參數,為「App $app」

    $params = $reflect->getParameters();

    $args   = [];

    foreach ($params as $param) {

        $name      = $param->getName();

        $lowerName = self::parseName($name);

        $class     = $param->getClass();

        //如果參數是一個類

        if ($class) {

            //將類型提示的參數實例化

            $args[] = $this->getObjectParam($class->getName(), $vars);

        } elseif (1 == $type && !empty($vars)) {

            $args[] = array_shift($vars);

        } elseif (0 == $type && isset($vars[$name])) {

            $args[] = $vars[$name];

        } elseif (0 == $type && isset($vars[$lowerName])) {

            $args[] = $vars[$lowerName];

        } elseif ($param->isDefaultValueAvailable()) {

            $args[] = $param->getDefaultValue();

        } else {

            throw new InvalidArgumentException('method param miss:' . $name);

        }

    }

    return $args;

}

 

而這之中,又最值得一提的是 getObjectParam() 方法:

protected function getObjectParam(string $className, array &$vars)

{

    $array = $vars;

    $value = array_shift($array);

    if ($value instanceof $className) {

        $result = $value;

        array_shift($vars);

    } else {

        //實例化傳入的類

        $result = $this->make($className);

    }

    return $result;

}

 

getObjectParam() 方法再一次光榮地調用 make() 方法,實例化一個類,而這個類,正是從 Http 的構造函數提取的參數,而這個參數又恰恰是一個類的實例 ——App 類的實例。到這裡,程式不僅通過 PHP 的反射類實例化了 Http 類,而且實例化了 Http 類的依賴 App 類。假如 App 類又依賴 C 類,C 類又依賴 D類…… 不管多少層,整個依賴鏈條依賴的類都可以實現實例化。

總的來說,整個過程大概是這樣的:需要實例化 Http 類 ==> 提取構造函數發現其依賴 App 類 ==> 開始實例化 App 類(如果發現還有依賴,則一直提取下去,直到天荒地老)==> 將實例化好的依賴(App 類的實例)傳入 Http 類來實例化 Http 類。

這個過程,起個裝逼的名字就叫做「依賴註入」,起個摸不著頭腦的名字,就叫做「控制反轉」。

這個過程,如果退回遠古時代,要實例化 Http 類,大概是這樣實現的(假如有很多層依賴):

.

.

.

$e = new E();

$d = new D($e);

$c = new D($d);

$app = new App($c);

$http = new Http($app);

.

.

.

 

這得有多累人。而現代 PHP,交給「容器」就好了。容器還有不少功能,後面再詳解。

以上就是ThinkPHP6源碼:從Http類的實例化看依賴註入是如何實現的的詳細內容。

更多PHP相關知識請關註我的專欄PHP​zhuanlan.zhihu.com


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 如圖所示先處理乘號和除號,再處理加減。 #include<bits/stdc++.h> using namespace std; bool res[101];int main(){ int n; cin>>n; int i,j,op1,op2; string inp; char op[3]; int ...
  • 實現紙牌游戲的隨機抽牌洗牌過程(item系列幾個內置方法的實例) 1、namedtuple:命名元組,可以創建一個沒有方法只有屬性的類 from collections import namedtuple card = namedtuple('card',['rank','suit']) # ran ...
  • 環境:xadmin django2.0 python3.7.4 操作登錄login()或者註銷logout()報以上錯誤的解決辦法如下: 在xadmin的views/website.py中 修改如下: from django.contrib.auth.views import login from ...
  • 問題現象 django xadmin中logout頁面在chrome瀏覽器中點擊關閉頁面無效,無法關閉相應的頁面 問題原因 高版本的chrome等瀏覽器不支持在window.colse()的寫法 問題源碼 在xadmin的templates的xadmin的views的logged_out.html中 ...
  • 問題描述 Alice和Bob正在玩井字棋游戲。 井字棋游戲的規則很簡單:兩人輪流往3*3的棋盤中放棋子,Alice放的是“X”,Bob放的是“O”,Alice執先。當同一種棋子占據一行、一列或一條對角線的三個格子時,游戲結束,該種棋子的持有者獲勝。當棋盤被填滿的時候,游戲結束,雙方平手。 Alice ...
  • 這幾個變數判斷函數在PHP開發中用的其實挺多的,而且粗看上去都差不多,但其實還是有不少的區別的,如果搞不清楚,也許就會遺留一些潛在的bug, 包括我自已也遇到過這樣的坑,比如有一次我就遇到過用empty判斷出現的問題,前端是允許輸入0的,但是我用empty判斷的話就為真,我就直接給報錯了,所以這裡就 ...
  • 需求場景 不同終端(PC端、手機端、平板),不同界面(列表頁、詳情頁),對圖片大小的要求不一樣, 如果所有場景下都使用同一尺寸的圖片,勢必對會網路帶寬及伺服器性能造成一定的影響,由此需要伺服器端能夠根據前端的請求參數,自動匹配出相對應的圖片資源,以此來降低服務端的壓力,同時也能給用戶帶來更友好的用戶 ...
  • ThinkPHP6 源碼分析之應用初始化 官方群點擊此處。 App Construct 先來看看在 __construct 中做了什麼,基本任何框架都會在這裡做一些基本的操作,也就是從這裡開始延伸出去。 public function __construct(string $rootPath = ' ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...