ThinkPHP6 核心分析(二):Request 类的实例化

说明#

修改日志:2019-10-28 更新到 6.0 正式版

接上一篇,得到 Http 类的一个实例后,程序接下来执行 $response = $http->run();。其中 run() 方法代码如下:

public function run(Request $request = null): Response
{
    //自动创建request对象
    $request = $request ?? $this->app->make('request', [], true);
    // 将Request类的实例保存到「$instances」数组
    $this->app->instance('request', $request);

    try {
        $response = $this->runWithRequest($request);
    } catch (Throwable $e) {
        $this->reportException($e);

        $response = $this->renderException($request, $e);
    }

    return $response;
}

从「request」标识找到要实例化的类#

run() 方法的第一行通过容器类实例 app 调用 make() 方法并传入 Request 类的标识来实例化 Request 类。具体过程如下分析。

通过 make() 方法首先解析得到 request 标识对应的标识 think\Request, 进一步递归解析,又得到 app\Request 类 —— 这个才是最终要实例化的类。
app\Request 类对应的文件位于 app 目录下,代码如下:

namespace app;

class Request extends \think\Request
{

}

实际上它啥事也没干,直接继承系统的 \think\Request。当然我们也可以在这里对系统的 Request 类进行改写重构。

调用 invokeClass () 方法#

从类的标识解析得到最终需要实例化的类(单例模式下,且该类还不存在实例)之后,程序调用 invokeClass () 方法,通过 PHP 的反射类实现类的实例化。由于 \think\Request 类存在__make() 方法,所以实例化之前首先调用该方法。__make() 方法代码如下:

public static function __make(App $app)
{
    //实例化自身
    $request = new static();

    // 保存超全局变量$_SERVER
    // 参考https://www.php.net/manual/zh/reserved.variables.server.php
    $request->server  = $_SERVER;

    // 跟前面的Http的实例化原理一样,实例化Env类并保存
    $request->env     = $app->env;

    $request->get     = $_GET;
    $request->post    = $_POST ?: $request->getInputData($request->input);
    $request->put     = $request->getInputData($request->input);
    $request->request = $_REQUEST;
    $request->cookie  = $_COOKIE;
    $request->file    = $_FILES ?? [];

    // 如果存在方法apache_request_headers则执行之
    // apache_request_headers的作用是获取所有HTTP请求头
    if (function_exists('apache_request_headers') && $result = apache_request_headers()) {
        $header = $result;
    } else {
        $header = [];
        $server = $_SERVER;
        foreach ($server as $key => $val) {
            if (0 === strpos($key, 'HTTP_')) {
                $key          = str_replace('_', '-', strtolower(substr($key, 5)));
                $header[$key] = $val;
            }
        }
        if (isset($server['CONTENT_TYPE'])) {
            $header['content-type'] = $server['CONTENT_TYPE'];
        }
        if (isset($server['CONTENT_LENGTH'])) {
            $header['content-length'] = $server['CONTENT_LENGTH'];
        }
    }

    //将数组的中所有KEY转为小写
    $request->header = array_change_key_case($header);
    //__make()方法最终返回Request类的实例
    return $request;
}

__make() 方法首先实例化 think\Request 类自身。think\Request 类构造函数如下:

public function __construct()
{
    // 保存 php://input
    //参考资料:http://www.nowamagic.net/academy/detail/12220520
    // php://input 用于读取POST数据
    //(可用于Coentent-Type取值为application/x-www-data-urlencoded、text/json、text/xml,
    // 不能用于multipart/form-data类型)
    //用$_POST的话,仅在Coentent-Type取值为application/x-www-data-urlencoded
    // 和multipart/form-data两种情况下有用
    $this->input = file_get_contents('php://input');
}

构造函数读取了 php://input 保存起来。接着,__make() 方法保存了一些请求相关的数据,最后返回一个 Request 类实例。最后的最后, make() 方法也成功得到该实例,整个过程跟 Http 类的实例化类似。该 Request 类实例部分成员变量如图:

ThinkPHP6 源码阅读(二):Request类是如何实例化的

保存「Request」类的实例到「$instances」数组#

得到 Request 类的实例后,run() 方法接着将该实例保存到「$instance」数组,以便后面单例模式要用到时可以直接获取。$instances 数组的值如图,Request 类的实例已保存在里面:

ThinkPHP6 源码阅读(二):Request类是如何实例化的

本作品采用《CC 协议》,转载必须注明作者和本文链接
Was mich nicht umbringt, macht mich stärker
本帖由系统于 3年前 自动加精