[TOC]
* * * * *
## 1 本节是应用启动文件(/thinkphp/library/think/App.php)的其他静态方法分析
**self::initModule()**
~~~
private static function initModule($module, $config)
{
$module = (COMMON_MODULE == $module || !APP_MULTI_MODULE) ? '' : $module . DS;
if (is_file(APP_PATH . $module . 'init' . EXT)) {
include APP_PATH . $module . 'init' . EXT;
} else {
$path = APP_PATH . $module;
$config = Config::load(APP_PATH . $module . 'config' . EXT);
if ($config['app_status']) {
$config = Config::load(APP_PATH . $module . $config['app_status'] . EXT);
}
if ($config['extra_config_list']) {
foreach ($config['extra_config_list'] as $name => $file) {
$file = strpos($file, '.') ? $file : $path . $file . EXT;
Config::load($file, is_string($name) ? $name : pathinfo($file, PATHINFO_FILENAME));
}
}
if (is_file($path . 'alias' . EXT)) {
Loader::addMap(include $path . 'alias' . EXT);
}
if (APP_HOOK && is_file($path . 'tags' . EXT)) {
Hook::import(include $path . 'tags' . EXT);
}
if (is_file($path . 'common' . EXT)) {
include $path . 'common' . EXT;
}
if ($config['lang_switch_on'] && $module) {
Lang::load($path . 'lang' . DS . LANG_SET . EXT);
}
}
}
~~~
**self::initModule()源码分析**
`$module = (COMMON_MODULE == $module || !APP_MULTI_MODULE) ? '' : $module . DS;`
模块目录名称获取
~~~
if (is_file(APP_PATH . $module . 'init' . EXT)) {
include APP_PATH . $module . 'init' . EXT;
}
~~~
检查是否存在模块对应的初始化入口文件init.php,见使用范例 模块开发
~~~
$path = APP_PATH . $module;
$config = Config::load(APP_PATH . $module . 'config' . EXT);
~~~
加载模块的配置文件 见使用范例 模块开发
~~~
if ($config['app_status']) {
$config = Config::load(APP_PATH . $module . $config['app_status'] . EXT);
}
~~~
加载模块状态配置文件 见使用范例 模块开发
~~~
if ($config['extra_config_list']) {
foreach ($config['extra_config_list'] as $name => $file) {
$file = strpos($file, '.') ? $file : $path . $file . EXT;
Config::load($file, is_string($name) ? $name : pathinfo($file, PATHINFO_FILENAME));
}
}
~~~
加载模块扩展配置文件 见使用范例模块开发
~~~
if (is_file($path . 'alias' . EXT)) {
Loader::addMap(include $path . 'alias' . EXT);
}
~~~
加载模块别名配置文件 见使用范例模块开发
~~~
if (APP_HOOK && is_file($path . 'tags' . EXT)) {
Hook::import(include $path . 'tags' . EXT);
}
~~~
加载模块行为扩展文件 见使用范例模块开发
~~~
if (is_file($path . 'common' . EXT)) {
include $path . 'common' . EXT;
}
~~~
加载模块公共函数文件 见使用范例模块开发
~~~
if ($config['lang_switch_on'] && $module) {
Lang::load($path . 'lang' . DS . LANG_SET . EXT);
}
~~~
加载模块语言包文件 见使用范例模块开发
* * * * *
* * * * *
**2 self::parsePathinfo()源代码**
~~~
private static function parsePathinfo(array $config)
{
if (isset($_GET[$config['var_pathinfo']])) {
$_SERVER['PATH_INFO'] = $_GET[$config['var_pathinfo']];
unset($_GET[$config['var_pathinfo']]);
} elseif (IS_CLI) {
// CLI模式下 index.php module/controller/action/params/...
$_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
}
APP_HOOK && Hook::listen('path_info');
if (!isset($_SERVER['PATH_INFO'])) {
foreach ($config['pathinfo_fetch'] as $type) {
if (!empty($_SERVER[$type])) {
$_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type], $_SERVER['SCRIPT_NAME'])) ?
substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
break;
}
}
}
}
~~~
**self::parsePathinfo()源代码分析**
~~~
if (isset($_GET[$config['var_pathinfo']])) {
$_SERVER['PATH_INFO'] = $_GET[$config['var_pathinfo']];
unset($_GET[$config['var_pathinfo']]);
} elseif (IS_CLI) {
// CLI模式下 index.php module/controller/action/params/...
$_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
}
~~~
检查$_GET参数是否包含$config配置的兼容模式参数,
兼容模式参数见附:全局配置文件
如果有 设置$_SERVER['PATH_INFO']为$_GET中兼容模式参数对应的参数的值
并删除$_GET兼容模式参数
如果没有检查模式检查
检查是否是CLI模式 然后设置$_SERVER['PATH_INFO']为CLI的$_SERVER['argv'][1]参数
这里的PATH_INFO其实就是调用的文件名称与参数
` APP_HOOK && Hook::listen('path_info');`
path_info分析回调
~~~
if (!isset($_SERVER['PATH_INFO'])) {
foreach ($config['pathinfo_fetch'] as $type) {
if (!empty($_SERVER[$type])) {
$_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type], $_SERVER['SCRIPT_NAME'])) ?
substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
break;
}
}
}
~~~
如果上面获取PATH_INFO失败
那么从配置信息$config['pathinfp_fetch']中依次读取PATH_INFP信息
经过上面调用获取了全局变量$_SERVER['PATH_INFO']的值,这个值在下面的self::route()中会用到
* * * * *
* * * * *
**3 self::route()源代码**
~~~
public static function route(array $config)
{
self::parsePathinfo($config);
if (empty($_SERVER['PATH_INFO'])) {
$_SERVER['PATH_INFO'] = '';
define('__INFO__', '');
define('__EXT__', '');
} else {
$_SERVER['PATH_INFO'] = trim($_SERVER['PATH_INFO'], '/');
define('__INFO__', $_SERVER['PATH_INFO']);
define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'], PATHINFO_EXTENSION)));
if ($config['url_deny_suffix'] && preg_match('/\.(' . $config['url_deny_suffix'] . ')$/i', __INFO__)) {
throw new Exception('url suffix deny');
}
$_SERVER['PATH_INFO'] = preg_replace($config['url_html_suffix'] ? '/\.(' . trim($config['url_html_suffix'], '.') . ')$/i' : '/\.' . __EXT__ . '$/i', '', __INFO__);
}
$depr = $config['pathinfo_depr'];
$result = false;
if (APP_ROUTE_ON && !empty($config['url_route_on'])) {
if (!empty($config['route'])) {
Route::register($config['route']);
}
$result = Route::check($_SERVER['PATH_INFO'], $depr, !IS_CLI ? $config['url_domain_deploy'] : false);
if (APP_ROUTE_MUST && false === $result && $config['url_route_must']) {
throw new Exception('route not define ');
}
}
if (false === $result) {
$result = Route::parseUrl($_SERVER['PATH_INFO'], $depr);
}
self::dispatch($result);
}
~~~
**self::route()源代码分析**
`self::parsePathinfo($config);`
调用上面的self::parsePathinfo()获取$_SERVER['PATH_INFO']的值
~~~
$_SERVER['PATH_INFO'] = '';
define('__INFO__', '');
define('__EXT__', '');
~~~
如果$_SERVER['PATH_INFO']为空
设置`__INFO__ __EXT__`全局变量为空
~~~
$_SERVER['PATH_INFO'] = trim($_SERVER['PATH_INFO'], '/');
define('__INFO__', $_SERVER['PATH_INFO']);
define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'], PATHINFO_EXTENSION)));
if ($config['url_deny_suffix'] && preg_match('/\.(' . $config['url_deny_suffix'] . ')$/i', __INFO__)) {
throw new Exception('url suffix deny');
}
$_SERVER['PATH_INFO'] = preg_replace($config['url_html_suffix'] ? '/\.(' . trim($config['url_html_suffix'], '.') . ')$/i' : '/\.' . __EXT__ . '$/i', '', __INFO__);
~~~
如果$_SERVER['PATH_INFO']为空
设置`__INFO__ __EXT__`为$_SERVER['PATH_INFO']的相应值。
并检查是否禁用相应的后缀请求 见使用范例 禁用URL后缀
接着去除正常的Url后缀???待分析
`$depr = $config['pathinfo_depr'];`
获取配置的pathinfo_depr,
~~~
if (APP_ROUTE_ON && !empty($config['url_route_on'])) {
}
~~~
检查全局变量APP_ROUTE_ON 是否开启路由检测
全局变量设置见 [附:全局变量文件](http://www.kancloud.cn/zmwtp/tp5/119430)
如果开启
~~~
if (!empty($config['route'])) {
Route::register($config['route']);
}
~~~
将配置中的路由添加到全局路由定义
全局路由定义见 [附:全局路由文件](http://www.kancloud.cn/zmwtp/tp5/119438)
`$result = Route::check($_SERVER['PATH_INFO'], $depr, !IS_CLI ? $config['url_domain_deploy'] : false);`
检查路由 ???待分析
~~~
if (APP_ROUTE_MUST && false === $result && $config['url_route_must']) {
throw new Exception('route not define ');
}
~~~
路由检测结果分析
~~~
if (false === $result) {
$result = Route::parseUrl($_SERVER['PATH_INFO'], $depr);
}
~~~
路由url分析 见 [附:全局路由文件](http://www.kancloud.cn/zmwtp/tp5/119438)
`self::dispatch($result);`
注册路由分析结果到app调度类型$dispatch
调度类型的使用见 [应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
## 3 总结
**initModule() 模块的初始化实现**
在系统主流程App::run()中有两处调用,
一处是App::run()刚开始总的公共模块初始化
self::initModule(COMMON_MODULE,Config::get())
一处是App::module()中的应用模块初始化
self::initModule(MODULE_NAME, $config);
**parsePathinfo() 获取请求信息**
**route() 根据请求信息进行路由分派**
其他的函数包括
run() 应用启动流程 见 [应用启动文件](http://www.kancloud.cn/zmwtp/tp5/119426)
invokeFunction() 调度回调函数 见 [应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
invokeMethod() 调度回调方法 见 [应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
bindParams() 调度参数合成 见 [应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
module() 调度回调模块 见 [应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
dispathc() 调度注册见[应用调度分析](http://www.kancloud.cn/zmwtp/tp5/119428)
- 更新记录
- 概述
- 文件索引
- 函数索引
- 章节格式
- 框架流程
- 前:章节说明
- 主:(index.php)入口
- 主:(start.php)框架引导
- 主:(App.php)应用启动
- 主:(App.php)应用调度
- C:(Controller.php)应用控制器
- M:(Model.php)数据模型
- V:(View.php)视图对象
- 附:(App.php)应用启动
- 附:(base.php)全局变量
- 附:(common.php)模式配置
- 附:(convention.php)全局配置
- 附:(Loader.php)自动加载器
- 附:(Build.php)自动生成
- 附:(Hook.php)监听回调
- 附:(Route.php)全局路由
- 附:(Response.php)数据输出
- 附:(Log.php)日志记录
- 附:(Exception.php)异常处理
- 框架工具
- 另:(helper.php)辅助函数
- 另:(Cache.php)数据缓存
- 另:(Cookie.php)cookie操作
- 另:(Console.php)控制台
- 另:(Debug.php)开发调试
- 另:(Error.php)错误处理
- 另:(Url.php)Url操作文件
- 另:(Loader.php)加载器实例化
- 另:(Input.php)数据输入
- 另:(Lang.php)语言包管理
- 另:(ORM.php)ORM基类
- 另:(Process.php)进程管理
- 另:(Session.php)session操作
- 另:(Template.php)模板解析
- 框架驱动
- D:(\config)配置解析
- D:(\controller)控制器扩展
- D:(\model)模型扩展
- D:(\db)数据库驱动
- D:(\view)模板解析
- D:(\template)模板标签库
- D:(\session)session驱动
- D:(\cache)缓存驱动
- D:(\console)控制台
- D:(\process)进程扩展
- T:(\traits)Trait目录
- D:(\exception)异常实现
- D:(\log)日志驱动
- 使用范例
- 服务器与框架的安装
- 控制器操作
- 数据模型操作
- 视图渲染控制
- MVC开发初探
- 模块开发
- 入口文件定义全局变量
- 运行模式开发
- 框架配置
- 自动生成应用
- 事件与插件注册
- 路由规则注册
- 输出控制
- 多种应用组织
- 综合应用
- tp框架整合后台auto架构快速开发
- 基础原理
- php默认全局变量
- php的魔术方法
- php命名空间
- php的自动加载
- php的composer
- php的反射
- php的trait机制
- php设计模式
- php的系统时区
- php的异常错误
- php的输出控制
- php的正则表达式
- php的闭包函数
- php的会话控制
- php的接口
- php的PDO
- php的字符串操作
- php的curl
- 框架心得
- 心:整体结构
- 心:配置详解
- 心:加载器详解
- 心:输入输出详解
- 心:url路由详解
- 心:模板详解
- 心:模型详解
- 心:日志详解
- 心:缓存详解
- 心:控制台详解
- 框架更新
- 4.20(验证类,助手函数)
- 4.27(新模型Model功能)
- 5.4(新数据库驱动)
- 7.28(自动加载)