[提問交流]ThinkPHP 框架執行應用的標準流程分析

發表於2020-04-04
總體來說,應用的流程涉及到幾個檔案:
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.        日誌資訊儲存寫入
回覆

相關文章