[转]ThinkPHP6.0核心分析--Request类的初始化
in ThinkPHP with 0 comment

[转]ThinkPHP6.0核心分析--Request类的初始化

in ThinkPHP with 0 comment

说明

接上一篇,得到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类实例部分成员变量如图: 转thinkphp4.png

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

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