總體來說,應用的流程涉及到幾個文件:
Index.php
ThinkPHP.php
Think.class.php
App.class.php
Dispatcher.class.php
ThinkPHP/Mode/common.php
ReadHtmlBehavior.class.php
Route.class.php
Hook.class.php
ContentReplaceBehavior.class.php
WriteHtmlCacheBehavior.class.php
ThinkPHP框架開發的應用的標准執行流程如下:
1. 用戶URL請求
2. 調用應用入口文件(通常是網站的index.php)
3. 載入框架入口文件(ThinkPHP.php)
4. 記錄初始運行時間和內存開銷
(引用自ThinkPHP.php) // 記錄開始運行時間
$GLOBALS['_beginTime'] = microtime(TRUE); // 記錄內存初始使用
define('MEMORY_LIMIT_ON',function_exists('memory_get_usage')); if(MEMORY_LIMIT_ON) $GLOBALS['_startUseMems'] = memory_get_usage();
5. 系統常量判斷及定義
(引用自ThinkPHP.php) // 系統常量定義
defined('THINK_PATH') or define('THINK_PATH', __DIR__.'/'); defined('APP_PATH') or define('APP_PATH', dirname($_SERVER['SCRIPT_FILENAME']).'/'); defined('APP_STATUS') or define('APP_STATUS', ''); // 應用狀態 加載對應的配置文件
defined('APP_DEBUG') or define('APP_DEBUG', false); // 是否調試模式
6. 載入框架引導類(Think\Think)並執行Think::start方法進行應用初始化
(引用自ThinkPHP.php) // 應用初始化
Think\Think::start();
7. 設置錯誤處理機制和自動加載機制
(引用Think.class.php) // 注冊AUTOLOAD方法
spl_autoload_register('Think\Think::autoload'); // 設定錯誤和異常處理
register_shutdown_function('Think\Think::fatalError'); set_error_handler('Think\Think::appError'); set_exception_handler('Think\Think::appException');
8. 調用Think\Storage類進行存儲初始化(由STORAGE_TYPE常量定義存儲類型)
(引用Think.class.php) // 初始化文件存儲方式
Storage::connect(STORAGE_TYPE);
9. 部署模式下如果存在應用編譯緩存文件則直接加載(直接跳轉到步驟22)
(引用Think.class.php) if(!APP_DEBUG && Storage::has($runtimefile)){ Storage::load($runtimefile); }
10. 讀取應用模式(由APP_MODE常量定義)的定義文件(以下以普通模式為例說明)
Thinkphp框架默認的應用模式 為普通模式。
-
(引用Think.class.php)
// 讀取應用模式 $mode = include is_file(CONF_PATH.'core.php')?CONF_PATH.'core.php':MODE_PATH.APP_MODE.'.php';
11. 加載當前應用模式定義的核心文件(普通模式是 ThinkPHP/Mode/common.php)
-
(common.php) THINK_PATH.'Conf/convention.php', // 系統慣例配置 CONF_PATH.'config'.CONF_EXT, // 應用公共配置
12. 加載慣例配置文件(普通模式是 ThinkPHP/Conf/convention.php)
13. 加載應用配置文件(普通模式是 Application/Common/Conf/config.php)
14. 加載系統別名定義
-
(common.php) // 別名定義 'alias' => array( 'Think\Log' => CORE_PATH . 'Log'.EXT, 'Think\Log\Driver\File' => CORE_PATH . 'Log/Driver/File'.EXT, 'Think\Exception' => CORE_PATH . 'Exception'.EXT, 'Think\Model' => CORE_PATH . 'Model'.EXT, 'Think\Db' => CORE_PATH . 'Db'.EXT, 'Think\Template' => CORE_PATH . 'Template'.EXT, 'Think\Cache' => CORE_PATH . 'Cache'.EXT, 'Think\Cache\Driver\File' => CORE_PATH . 'Cache/Driver/File'.EXT, 'Think\Storage' => CORE_PATH . 'Storage'.EXT, ),
15. 判斷並讀取應用別名定義文件(普通模式是 Application/Common/Conf/alias.php)
16. 加載系統行為定義
17. 判斷並讀取應用行為定義文件(普通模式是 Application/Common/Conf/tags.php)
-
(tags.php) 'app_init'=>array('Common\Behavior\InitHookBehavior')
18. 加載框架底層語言包(普通模式是 ThinkPHP/Lang/zh-cn.php)
19. 如果是部署模式則生成應用編譯緩存文件
20. 加載調試模式系統配置文件(ThinkPHP/Conf/debug.php)
21. 判斷並讀取應用的調試配置文件(默認是 Application/Common/Conf/debug.php)
22. 判斷應用狀態並讀取狀態配置文件(如果APP_STATUS常量定義不為空的話)
-
(think.class.php) // 讀取當前應用狀態對應的配置文件 if(APP_STATUS && is_file(CONF_PATH.APP_STATUS.CONF_EXT)) C(include CONF_PATH.APP_STATUS.CONF_EXT);
23. 檢測應用目錄結構並自動生成(如果CHECK_APP_DIR配置開啟並且RUNTIME_PATH目錄不存在的情況下)
-
think.class.php // 檢查應用目錄結構 如果不存在則自動創建 if(C('CHECK_APP_DIR')) { $module = defined('BIND_MODULE') ? BIND_MODULE : C('DEFAULT_MODULE'); if(!is_dir(APP_PATH.$module) || !is_dir(LOG_PATH)){ // 檢測應用目錄結構 Build::checkDir($module); } }
24. 調用Think\App類的run方法啟動應用
-
think.class.php // 運行應用 App::run();
25. 應用初始化(app_init)標簽位偵聽並執行綁定行為
26. 判斷並加載動態配置和函數文件
27. 調用Think\Dispatcher::dispatch方法進行URL請求調度
-
App.class.php Dispatcher::dispatch();
28. 自動識別兼容URL模式和命令行模式下面的$_SERVER['PATH_INFO']參數
-
Dispatcher.class.php $_SERVER['PATH_INFO'] = $_GET[$varPath];
29. 檢測域名部署以及完成模塊和控制器的綁定操作(APP_SUB_DOMAIN_DEPLOY參數開啟)
-
Dispatcher.class.php
30. 分析URL地址中的PATH_INFO信息
-
Dispatcher.class.php
31. 獲取請求的模塊信息
32. 檢測模塊是否存在和允許訪問
33. 判斷並加載模塊配置文件、別名定義、行為定義及函數文件
34. 判斷並加載模塊的動態配置和函數文件
35. 模塊的URL模式判斷
36. 模塊的路由檢測(URL_ROUTER_ON開啟)
-
Dispatcher.class.php
37. PATH_INFO處理(path_info)標簽位偵聽並執行綁定行為
38. URL后綴檢測(URL_DENY_SUFFIX以及URL_HTML_SUFFIX處理)
39. 獲取當前控制器和操作,以及URL其他參數
40. URL請求調度完成(url_dispatch)標簽位偵聽並執行綁定行為
41. 應用開始(app_begin)標簽位偵聽並執行綁定行為
-
App.class.php static public function run() { // 應用初始化標簽 Hook::listen('app_init'); App::init(); // 應用開始標簽 Hook::listen('app_begin');
42. 調用SESSION_OPTIONS配置參數進行Session初始化(如果不是命令行模式)
-
// Session初始化 if(!IS_CLI){ session(C('SESSION_OPTIONS')); }
43. 根據請求執行控制器方法
44. 如果控制器不存在則檢測空控制器是否存在
45. 控制器開始(action_begin)標簽位偵聽並執行綁定行為
-
Controller.class.php public function __construct() { Hook::listen('action_begin',$this->config); //實例化視圖類 $this->view = Think::instance('Think\View'); //控制器初始化 if(method_exists($this,'_initialize')) $this->_initialize(); }
46. 默認調用系統的ReadHtmlCache行為讀取靜態緩存(HTML_CACHE_ON參數開啟)
47. 判斷並調用控制器的_initialize初始化方法
-
Controller.class.php if(method_exists($this,'_initialize')) $this->_initialize();
48. 判斷操作方法是否存在,如果不存在則檢測是否定義空操作方法
49. 判斷前置操作方法是否定義,有的話執行
50. Action參數綁定檢測,自動匹配操作方法的參數
51. 如果有模版渲染(調用控制器display方法)
52. 視圖開始(view_begin)標簽位偵聽並執行綁定行為
53. 調用Think\View的fetch方法解析並獲取模版內容
-
View.class.php
54. 自動識別當前主題以及定位模版文件
55. 視圖解析(view_parse)標簽位偵聽並執行綁定行為
-
View.class.php Hook::listen('view_parse',$params); common.php 'view_parse' => array( 'Behavior\ParseTemplateBehavior', // 模板解析 支持PHP、內置模板引擎和第三方模板引擎 ),
56. 默認調用內置ParseTemplate行為解析模版(普通模式下面)
-
View.class.php public function parseTemplate($template='')
57. 模版引擎解析模版內容后生成模版緩存
58. 模版過濾替換(template_filter)標簽位偵聽並執行綁定行為
-
Template.class.php Hook::listen('template_filter',$tmplContent);
59. 默認調用系統的ContentReplace行為進行模版替換
-
'template_filter'=> array( 'Behavior\ContentReplaceBehavior', // 模板輸出替換 ), ContentReplaceBehavior.class.php class ContentReplaceBehavior {
60. 輸出內容過濾(view_filter)標簽位偵聽並執行綁定行為
-
'view_filter' => array( 'Behavior\WriteHtmlCacheBehavior', // 寫入靜態緩存 ),
61. 默認調用系統的WriteHtmlCache行為寫入靜態緩存(HTML_CACHE_ON參數開啟)
-
WriteHtmlCacheBehavior.class.php class WriteHtmlCacheBehavior {
62. 調用Think\View類的render方法輸出渲染內容
63. 視圖結束(view_end)標簽位偵聽並執行綁定行為
view.class.php
64. 判斷后置操作方法是否定義,有的話執行
65. 控制器結束(action_end)標簽位偵聽並執行綁定行為
-
Controller.class.php
66. 應用結束(app_end)標簽位偵聽並執行綁定行為
-
App.class.php Hook::listen('app_end');
67. 執行系統的ShowPageTrace行為(SHOW_PAGE_TRACE參數開啟並且不是AJAX請求)
68. 日志信息存儲寫入