一、加載中間件
上一篇分析了應用的初始化,也就是對 Http 類的 run() 方法里面調用的 runWithRequest () 方法的第一行代碼 $this->initialize() 的展開分析。讓我們再看一眼 runWithRequest () 方法的前幾行:
protected function runWithRequest(Request $request)
{
$this->initialize();
// 加載全局中間件
$this->loadMiddleware();
.
.
.
應用初始化后,接下來開始處理中間件。
中間件類的初始化
loadMiddleware 方法:
protected function loadMiddleware(): void
{
if (is_file($this->app->getBasePath() . 'middleware.php')) {
$this->app->middleware->import(include $this->app->getBasePath() . 'middleware.php');
}
}
依然是百用不厭的套路,通過 $this->app->middleware 來實例化中間件并獲取其實例。
導入中間件
通過 $this->app->middleware 得到 Middleware 類的實例后,接著程序調用 import 方法,傳入從「app」目錄下的「middleware.php」文件中讀取的數據。該文件的原始內容如下(原來全部注釋掉的):
return [
// 全局請求緩存
// \think\middleware\CheckRequestCache::class,
// 多語言加載
\think\middleware\LoadLangPack::class,
// Session初始化
// \think\middleware\SessionInit::class,
// 頁面Trace調試
\think\middleware\TraceDebug::class,
];
這里為了研究中間件是如何加載的,先去掉兩個注釋,也就是添加兩個中間件。接下來看 import 方法:
public function import(array $middlewares = [], string $type = 'global'): void
{
foreach ($middlewares as $middleware) {
$this->add($middleware, $type);
}
}
該方法傳入一個中間件的數組和一個中間件類型,默認為 global,關鍵是里面的 add 方法。跳到 add 方法:
public function add($middleware, string $type = 'route'): void
{
if (is_null($middleware)) {
return;
}
$middleware = $this->buildMiddleware($middleware, $type);
if ($middleware) {
$this->queue[$type][] = $middleware;
// 去除重復
$this->queue[$type] = array_unique($this->queue[$type], SORT_REGULAR);
}
}
實際上真正干活的是 buildMiddleware 方法,直接前往:
protected function buildMiddleware($middleware, string $type): array
{
// 是否是數組
if (is_array($middleware)) {
// 列出中間件及其參數
// 這里說明我們可以給中間件傳入參數,且形式為 [中間件, 參數]
list($middleware, $param) = $middleware;
}
// 是否是一個閉包
// 說明中間件可以是一個閉包
if ($middleware instanceof \Closure) {
//返回閉包和參數
return [$middleware, $param ?? null];
}
// 排除了上面幾種類型,且不是字符串,拋出錯誤
if (!is_string($middleware)) {
throw new InvalidArgumentException('The middleware is invalid');
}
//中間件別名檢查
$alias = $this->app->config->get('middleware.alias', []);
if (isset($alias[$middleware])) {
$middleware = $alias[$middleware];
}
//如果中間件有包含中間件(說明中間件可以嵌套)
//再走一遍「import」遞歸解析
if (is_array($middleware)) {
$this->import($middleware, $type);
return [];
}
//返回解析結果
return [[$middleware, 'handle'], $param ?? null];
}
詳細分析見以上代碼注釋。最后返回的結果,在 add 方法中,執行 $ this->queue[$type][] = $middleware; 添加到一個隊列。最終的解析結果大概是這樣的(app/middleware.php 去掉部分中間件的注釋):
在這里插入圖片描述
至此,全局中間件就加載完畢。
二、多應用解析
加載完中間件,接下來一步是多應用解析(ThinkPHP 6 開始支持多應用模式)。
if ($this->multi) {
$this->parseMultiApp();
}
1
2
3
注意到,Http 類的構造函數:
public function __construct(App $app)
{
$this->app = $app;
//多應用解析,通過判斷「app」目錄下有無「controller」目錄,沒有就是多應用模式
$this->multi = is_dir($this->app->getBasePath() . 'controller') ? false : true;
}
可以看到,程序是通過判斷「app」目錄下有無「controller」目錄來決定是否是多應用模式的。
接著看主要方法 parseMultiApp:
protected function parseMultiApp(): void
{
// 雖然在「Http」的構造函數自動判斷過是否開啟多應用
//如果沒有controller目錄,$this->multi為true,就會來到本方法
// 接著還要看配置文件是否有配置
if ($this->app->config->get('app.auto_multi_app', false)) {
// 自動多應用識別
$this->bindDomain = false;
// 獲取域名綁定
$bind = $this->app->config->get('app.domain_bind', []);
// 如果有域名綁定
if (!empty($bind)) {
// 獲取當前子域名
$subDomain = $this->app->request->subDomain();
$domain = $this->app->request->host(true);
//完整域名綁定
if (isset($bind[$domain])) {
$appName = $bind[$domain];
$this->bindDomain = true;
//子域名綁定
} elseif (isset($bind[$subDomain])) {
$appName = $bind[$subDomain];
$this->bindDomain = true;
//二級泛域名綁定
} elseif (isset($bind['*'])) {
$appName = $bind['*'];
$this->bindDomain = true;
}
}
//如果沒有域名綁定
if (!$this->bindDomain) {
//獲取別名映射
$map = $this->app->config->get('app.app_map', []);
//獲取禁止URL訪問目錄
$deny = $this->app->config->get('app.deny_app_list', []);
//獲取當前請求URL的pathinfo信息(含URL后綴)
// 比如 index/index/index
$path = $this->app->request->pathinfo();
// 比如,從index/index/index獲取得index
$name = current(explode('/', $path));
//解析別名映射
if (isset($map[$name])) {
//如果這個別名映射到的是一個閉包
//這樣不知有啥用
if ($map[$name] instanceof Closure) {
$result = call_user_func_array($map[$name], [$this]);
$appName = $result ?: $name;
//直接取得應用名
} else {
$appName = $map[$name];
}
//$name不為空且$name在$map數組中作為KEY,或者$name是禁止URL方位的目錄
} elseif ($name && (false !== array_search($name, $map) || in_array($name, $deny))) {
throw new HttpException(404, 'app not exists:' . $name);
} elseif ($name && isset($map['*'])) {
$appName = $map['*'];
} else {
$appName = $name;
}
if ($name) {
$this->app->request->setRoot('/' . $name);
$this->app->request->setPathinfo(strpos($path, '/') ? ltrim(strstr($path, '/'), '/') : '');
}
}
} else {
$appName = $this->name ?: $this->getScriptName();
}
$this->loadApp($appName ?: $this->app->config->get('app.default_app', 'index'));
}
可以看到,「pathinfo」信息的第一節會被解析成應用名稱,比如 index/index/index/ 中的 index。方法的最后還調用了 loadApp 方法,執行的操作與前面應用的初始化類似,只是加載的文件都在該應用的目錄。
跟之前的版本對比,ThinkPHP 6 貌似把原先的模塊改造成了多應用,因為多應用情況下,應用名跟之前的模塊名都是從 pathinfo 的第一節解析出來的,新的文檔也沒見到模塊的內容了。
猜你喜歡
聯絡方式:
深圳市龍華區觀瀾街道黎光社區黎光新工業區
電話:400-092-7217
郵箱:szyiqiwl@163.com
QQ:3275375818
-
電商系統網站的網絡營銷要如何去做?
電商系統網站的網絡營銷要怎么去做是很多人都在關心的問題,電商系統的開發跟開通,是電子商務時代的發展趨勢。企業商家為了風口上的市場利潤
-
營銷型網站如何才能吸引用戶?
作為網站建設中一個最具營銷潛力的網站,營銷型網站的注定比其他類型網站能更加吸引人群,營銷效果也將是最顯著的。那么為了讓營銷型
-
做營銷型網站如何提升用戶吸引力?
相信許多企業在進行營銷型網站建設時,最擔心的便是網站優化對用戶的吸引力不強,這樣用戶對網站就會失去興致,從而失去用戶支持
-
為什么有些網站長期不更新,排名卻靠前?
只要是站長都知道網站seo優化,都知道做優化最關鍵的就是網站更新,但是可能你看到很多網站長期的不更新,人家的網站推廣排名一樣的靠前
-
為何網絡營銷推廣對流量如此重要呢?
網絡營銷推廣是一種常見的營銷活動,而這種活動往往在網上存在的。大家在現實中的營銷推廣需要帶來的則是人的流量。這里的網絡營銷推廣