🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
[TOC] # 说明 接上一篇,得到`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()方法实例化Request类的过程分析 从类的标识解析得到最终需要实例化的类(单例模式下,且该类还不存在实例)之后,程序调用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`类实例部分成员变量如图: ![](https://img.kancloud.cn/7b/f0/7bf0306e590bf61967b7da701512229e_315x594.PNG) # 保存「Request」类的实例到「$instances」数组 得到`Request`类的实例后,`run()`方法接着将该实例保存到「$instance」数组,以便后面单例模式要用到时可以直接获取。`$instances`数组的值如图,`Request`类的实例已保存在里面: ![](https://img.kancloud.cn/76/30/76308643eadfa8ce031e67e7d7831416_340x130.PNG)