转载请注明出处,谢谢。
框架目录
3.2的版本
- |—ThinkPHP框架目录(能将其部署在非Web目录下)
- |——Common 核心公共函数
- |——Conf 核心配置目录
- |——Lang 核心语言包目录
- |——Library 框架类库目录
- |———Think 核心Think类库包目录
- |———Behavior 行为类库目录
- |———Org Org类库包目录
- |———Vendor 第三方类库目录
- |———… 更多类库目录
- |——Extend 框架扩展目录
- |——Tpl 系统模板目录
- |——LICENSE.txt 系统类库映射文件
- |——logo.png 框架LOGO文件
- |——README.txt 框架README文件
- |——index.php 框架入口文件
应用入口文件和Public资源目录外,其他文件都放到非WEB目录下面,具有更好的安全性。
系统流程
- 用户URL请求
- 调用应用入口文件(通常是网站的index.php)
- 载入框架入口文件(ThinkPHP.php)
- 记录初始运行时间和内存开销
- 系统常量判断及定义
- 载入框架引导类(Think\Think)并执行Think::start方法进行应用初始化
<?php
// +----------------------------------------------------------------------
// | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
// +----------------------------------------------------------------------
// | Copyright (c) 2006-2014 http://thinkphp.cn All rights reserved.
// +----------------------------------------------------------------------
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
// +----------------------------------------------------------------------
// | Author: liu21st <liu21st@gmail.com>
// +----------------------------------------------------------------------
//----------------------------------
// ThinkPHP公共入口文件
//----------------------------------
// 记录开始运行时间
$GLOBALS['_beginTime'] = microtime(TRUE);
// 记录内存初始使用
define('MEMORY_LIMIT_ON',function_exists('memory_get_usage'));
if(MEMORY_LIMIT_ON) $GLOBALS['_startUseMems'] = memory_get_usage();
// 版本信息
const THINK_VERSION = '3.2.2';
// URL 模式定义
const URL_COMMON = 0; //普通模式
const URL_PATHINFO = 1; //PATHINFO模式
const URL_REWRITE = 2; //REWRITE模式
const URL_COMPAT = 3; // 兼容模式
// 类文件后缀
const EXT = '.class.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); // 是否调试模式
if(function_exists('saeAutoLoader')){// 自动识别SAE环境
defined('APP_MODE') or define('APP_MODE', 'sae');
defined('STORAGE_TYPE') or define('STORAGE_TYPE', 'Sae');
}else{
defined('APP_MODE') or define('APP_MODE', 'common'); // 应用模式 默认为普通模式
defined('STORAGE_TYPE') or define('STORAGE_TYPE', 'File'); // 存储类型 默认为File
}
defined('RUNTIME_PATH') or define('RUNTIME_PATH', APP_PATH.'Runtime/'); // 系统运行时目录
defined('LIB_PATH') or define('LIB_PATH', realpath(THINK_PATH.'Library').'/'); // 系统核心类库目录
defined('CORE_PATH') or define('CORE_PATH', LIB_PATH.'Think/'); // Think类库目录
defined('BEHAVIOR_PATH')or define('BEHAVIOR_PATH', LIB_PATH.'Behavior/'); // 行为类库目录
defined('MODE_PATH') or define('MODE_PATH', THINK_PATH.'Mode/'); // 系统应用模式目录
defined('VENDOR_PATH') or define('VENDOR_PATH', LIB_PATH.'Vendor/'); // 第三方类库目录
defined('COMMON_PATH') or define('COMMON_PATH', APP_PATH.'Common/'); // 应用公共目录
defined('CONF_PATH') or define('CONF_PATH', COMMON_PATH.'Conf/'); // 应用配置目录
defined('LANG_PATH') or define('LANG_PATH', COMMON_PATH.'Lang/'); // 应用语言目录
defined('HTML_PATH') or define('HTML_PATH', APP_PATH.'Html/'); // 应用静态目录
defined('LOG_PATH') or define('LOG_PATH', RUNTIME_PATH.'Logs/'); // 应用日志目录
defined('TEMP_PATH') or define('TEMP_PATH', RUNTIME_PATH.'Temp/'); // 应用缓存目录
defined('DATA_PATH') or define('DATA_PATH', RUNTIME_PATH.'Data/'); // 应用数据目录
defined('CACHE_PATH') or define('CACHE_PATH', RUNTIME_PATH.'Cache/'); // 应用模板缓存目录
defined('CONF_EXT') or define('CONF_EXT', '.php'); // 配置文件后缀
defined('CONF_PARSE') or define('CONF_PARSE', ''); // 配置文件解析方法
// 系统信息
if(version_compare(PHP_VERSION,'5.4.0','<')) {
ini_set('magic_quotes_runtime',0);
define('MAGIC_QUOTES_GPC',get_magic_quotes_gpc()?True:False);
}else{
define('MAGIC_QUOTES_GPC',false);
}
define('IS_CGI',(0 === strpos(PHP_SAPI,'cgi') || false !== strpos(PHP_SAPI,'fcgi')) ? 1 : 0 );
define('IS_WIN',strstr(PHP_OS, 'WIN') ? 1 : 0 );
define('IS_CLI',PHP_SAPI=='cli'? 1 : 0);
if(!IS_CLI) {
// 当前文件名
if(!defined('_PHP_FILE_')) {
if(IS_CGI) {
//CGI/FASTCGI模式下
$_temp = explode('.php',$_SERVER['PHP_SELF']);
define('_PHP_FILE_', rtrim(str_replace($_SERVER['HTTP_HOST'],'',$_temp[0].'.php'),'/'));
}else {
define('_PHP_FILE_', rtrim($_SERVER['SCRIPT_NAME'],'/'));
}
}
if(!defined('__ROOT__')) {
$_root = rtrim(dirname(_PHP_FILE_),'/');
define('__ROOT__', (($_root=='/' || $_root=='\\')?'':$_root));
}
}
// 加载核心Think类
require CORE_PATH.'Think'.EXT;
// 应用初始化
Think\Think::start();
- 设置错误处理机制和自动加载机制
- 调用Think\Storage类进行存储初始化(由STORAGE_TYPE常量定义存储类型)
- 部署模式下如果存在应用编译缓存文件则直接加载(直接跳转到步骤22)
- 读取应用模式(由APP_MODE常量定义)的定义文件(以下以普通模式为例说明)
- 加载当前应用模式定义的核心文件(普通模式是 ThinkPHP/Conf/Mode/common.php)
- 加载惯例配置文件(普通模式是 ThinkPHP/Conf/convention.php)
- 加载应用配置文件(普通模式是 Application/Common/Conf/config.php)
- 加载系统别名定义
- 判断并读取应用别名定义文件(普通模式是 Application/Common/Conf/alias.php)
- 加载系统行为定义
- 判断并读取应用行为定义文件(普通模式是 Application/Common/Conf/tags.php)
- 加载框架底层语言包(普通模式是 ThinkPHP/Lang/zh-cn.php)
- 如果是部署模式则生成应用编译缓存文件
- 加载调试模式系统配置文件(ThinkPHP/Conf/debug.php)
- 判断并读取应用的调试配置文件(默认是 Application/Common/Conf/debug.php)
- 判断应用状态并读取状态配置文件(如果APP_STATUS常量定义不为空的话)
- 检测应用目录结构并自动生成(如果CHECK_APP_DIR配置开启并且RUNTIME_PATH目录不存在的情况下)
- 调用Think\App类的run方法启动应用
- 应用初始化(app_init)标签位侦听并执行绑定行为
- 判断并加载动态配置和函数文件
- 调用Think\Dispatcher::dispatch方法进行URL请求调度
- 自动识别兼容URL模式和命令行模式下面的$_SERVER[‘PATH_INFO’]参数
- 检测域名部署以及完成模块和控制器的绑定操作(APP_SUB_DOMAIN_DEPLOY参数开启)
- 分析URL地址中的PATH_INFO信息
- 获取请求的模块信息
- 检测模块是否存在和允许访问
- 判断并加载模块配置文件、别名定义、行为定义及函数文件
- 判断并加载模块的动态配置和函数文件
- 模块的URL模式判断
- 模块的路由检测(URL_ROUTER_ON开启)
- PATH_INFO处理(path_info)标签位侦听并执行绑定行为
- URL后缀检测(URL_DENY_SUFFIX以及URL_HTML_SUFFIX处理)
- 获取当前控制器和操作,以及URL其他参数
- URL请求调度完成(url_dispatch)标签位侦听并执行绑定行为
- 应用开始(app_begin)标签位侦听并执行绑定行为
- 调用SESSION_OPTIONS配置参数进行Session初始化(如果不是命令行模式)
- 根据请求执行控制器方法
- 如果控制器不存在则检测空控制器是否存在
- 控制器开始(action_begin)标签位侦听并执行绑定行为
- 默认调用系统的ReadHtmlCache行为读取静态缓存(HTML_CACHE_ON参数开启)
- 判断并调用控制器的_initialize初始化方法
- 判断操作方法是否存在,如果不存在则检测是否定义空操作方法
- 判断前置操作方法是否定义,有的话执行
- Action参数绑定检测,自动匹配操作方法的参数
- 如果有模版渲染(调用控制器display方法)
- 视图开始(view_begin)标签位侦听并执行绑定行为
- 调用Think\View的fetch方法解析并获取模版内容
- 自动识别当前主题以及定位模版文件
- 视图解析(view_parse)标签位侦听并执行绑定行为
- 默认调用内置ParseTemplate行为解析模版(普通模式下面)
- 模版引擎解析模版内容后生成模版缓存
- 模版过滤替换(template_filter)标签位侦听并执行绑定行为
- 默认调用系统的ContentReplace行为进行模版替换
- 输出内容过滤(view_filter)标签位侦听并执行绑定行为
- 默认调用系统的WriteHtmlCache行为写入静态缓存(HTML_CACHE_ON参数开启)
- 调用Think\View类的render方法输出渲染内容
- 视图结束(view_end)标签位侦听并执行绑定行为
- 判断后置操作方法是否定义,有的话执行
- 控制器结束(action_end)标签位侦听并执行绑定行为
- 应用结束(app_end)标签位侦听并执行绑定行为
- 执行系统的ShowPageTrace行为(SHOW_PAGE_TRACE参数开启并且不是AJAX请求)
- 日志信息存储写入
如果是部署模式下面的第二次请求的话,上面的流程中的步骤10( 10. 读取应用模式(由APP_MODE常量定义)的定义文件(以下以普通模式为例说明))~21可以省略。
16调用Think\Dispatcher::dispatch方法进行URL请求调度
<?php
/**
* 输出内容文本可以包括Html
* @access private
* @param string $content 输出内容
* @param string $charset 模板输出字符集
* @param string $contentType 输出类型
* @return mixed
*/
private function render($content,$charset='',$contentType=''){
if(empty($charset)) $charset = C('DEFAULT_CHARSET');
if(empty($contentType)) $contentType = C('TMPL_CONTENT_TYPE');
// 网页字符编码
header('Content-Type:'.$contentType.'; charset='.$charset);
header('Cache-control: '.C('HTTP_CACHE_CONTROL')); // 页面缓存控制
header('X-Powered-By:ThinkPHP');
// 输出模板文件
echo $content;
}