App類可以說是總框架的第二個真正入口了柬姚,所有的框架邏輯操作都在這里完成。這也是為什么我要把他提在其他核心組件類之前分析的原因帕识。我們先看入口文件start.php代碼:
require __DIR__ . '/base.php';
App::run()->send();
base.php在第一章里已經(jīng)說了本缠,做了前期的準備工作(定義系統(tǒng)必要的常量以及注冊自動加載),然后就執(zhí)行App的run然后send沙咏。這篇就說下一個請求從開始到結(jié)束的整個流程run做了什么,主要涉及到Request班套、route肢藐。也建議大家先看下base.php定義的常量以及config配置。這里是官方給出的更詳細的參考文檔:
常量定義:http://www.kancloud.cn/manual/thinkphp5/118137
配置參考:http://www.kancloud.cn/manual/thinkphp5/118136
PS:下文提到EXT均為文件后綴定義吱韭,TP為當前講述的框架
對app::run()做一個流程性的總結(jié)闡述:
流程代碼有做精簡窖壕,略去業(yè)務性的判斷以及更好展現(xiàn)流程代碼優(yōu)化
1.1、加載當前模塊(module)下的初始化文件({$module}init.EXT)在我理解杉女,(我覺得這個文件官方意圖是希望我們框架使用者將其他的一些功能性的配置文件以及自定義一些必要的內(nèi)容進行一個集合式的加載。當如如果這個文件不存在直接讀取其下面指定的文件)見代碼:
init($module = '') {
is_file($current_module . 'init' . EXT) {
//這玩意其實可以自定義的將下面這些東西整合起來打包
include $module . 'init' . EXT;
} else {
$config = Config::load($module . 'config' . CONF_EXT);
//app狀態(tài)環(huán)境的一個配置開發(fā)環(huán)境鸳吸、生產(chǎn)環(huán)境熏挎、測試環(huán)境等不同配置
if ($config['app_status']) {
$config = Config::load($module . $config['app_status'] . CONF_EXT);
}
//讀取擴展配置文件 當然你必須在$config里面定義了extra_config_list
foreache($config['extra_config_list'] as $config){
Config::load($module . $config . CONF_EXT);
}
//加載別名文件 $module . 'alias' . EXT
Loader::addClassMap(include $module . 'alias' . EXT);
//加載行為擴展文件
Hook::import($module . 'tags' . EXT);
//// 加載別名文件
include $module. 'common' . EXT;
//加載語言包
Lang::load( 'lang' . {$lang_type} . EXT);
}
return Config::get();
}
1.2、讀取配置里面的也就是1.1的配置命名空間相關配置晌砾、額外其他文件進行加載見代碼:
initCommon(){
$config = self::init();
//調(diào)試模式設置
self::$debug = Config::get('app_debug');
…………
//命名空間加載(這里說下這里加載是對命名空間自定義簡化或者其他想法進行的一個key=>value配置加載)
Loader::addNamespace($config['app_namespace'], APP_PATH);
//設置系統(tǒng)時區(qū)
date_default_timezone_set($config['default_timezone']);
self::$init = $config;
return self::$init;
}
2坎拐、根據(jù)請求URL以及配置config分析出調(diào)度信息(也就是當前操作狀態(tài)、模塊养匈、控制器哼勇、操作、參數(shù))
run(Request $request = null){
//單例個Request對象
is_null($request) && $request = Request::instance();
//這個就是 part1 了
$config = self::initCommon();
try {
//多語言
Lang::load(THINK_PATH . 'lang' . DS . $request->langset() . EXT);
//獲取調(diào)度信息 這個Request就是剛才的對象 準備去routeCheck里面干大事去
/*
routcheck返回的調(diào)度信息大致這樣的 當然type是正常請求controller/action 其他類型后講
$dispatch = [
'type' => 'module',
'module' => [
{$module}, {$controller}, {$action.$params}
]
]
*/
$check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];
……
}
}
routeCheck($request, array $config){
……
//Request::path獲取http $_SERVER以及根據(jù)config配置參數(shù)進行處理
/*
$path = '{$module}/{$controller}/{$action}呕乎?{$param1}={$val1}&{$param2}={$val2}……'
*/
$path = $request->path();
//pathinfo 定義的分隔符 去補習手冊配置吧
$depr = $config['pathinfo_depr'];
$result = false;
//路由檢測
$check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];
if($check) {
//路由配置導入
Route::import($config['route']);
//路由檢測 詳細見Route類源碼分析
$result = Route::check($request, $path, $depr, $config['url_domain_deploy']);
//強制走路由配置
$must = $config['url_route_must'];
if ($must && false === $result) {
//GG……route not found
}
}
if (false === $result) {
// 路由無效 解析模塊/控制器/操作/參數(shù)... 支持控制器自動搜索 就是對請求url進行按照規(guī)則進行整理 詳route類
$result = Route::parseUrl($path, $depr, $config['controller_auto_search']);
}
return $result;
}
3积担、進行分發(fā)調(diào)度執(zhí)行。
run(Request $request = null){
is_null($request) && $request = Request::instance();
$config = self::initCommon();
try {
Lang::load(THINK_PATH . 'lang' . DS . $request->langset() . EXT);
$dispatch = self::routeCheck($request, $config);
……
switch($dispatch['type']){
……
//進行請求調(diào)度
case 'module':
$data = self::module($dispatch['module'], $config, isset($dispatch['convert']) ? $dispatch['convert'] : null);
break;
……
default:
throw new \InvalidArgumentException('dispatch type not support');
}catch (HttpResponseException $exception) {
$data = $exception->getResponse();
}
Loader::clearInstance();
……
return $response;
}
}
下一章會著重App::module進行分析猬仁,會穿插Route相關內(nèi)容帝璧。