这周活比较轻松,正好过一遍thinkphp源码(3.1.2版),弄清楚该框架内部实现的流程,增长些知识。
第一步,安装xdebug,写个测试页,里面写上phpinfo();把内容复制贴到http://www.xdebug.org/find-binary.php页面里,点击提交,之后会提示本机php对应的xdebug版本,下载后放到php目录下的ext目录内,再把下面的配置写入php.ini内,之后重启apache服务器,再刷新测试页面,看界面上是否有xdebug字样。
- zend_extension="D:\server\php\php-5.3.8-Win32-VC9-x86\ext\php_xdebug-2.1.4-5.3-vc9.dll"
-
- [Xdebug]
-
- ;是否开启自动跟踪
- xdebug.auto_trace = On
- ;是否开启异常跟踪
- xdebug.show_exception_trace = On
- ;是否开启远程调试自动启动
- xdebug.remote_autostart = On
- ;是否开启远程调试
- xdebug.remote_enable = On
- ;允许调试的客户端IP
-
- ;远程调试的端口(默认9000)
- xdebug.remote_port=9001
- ;调试插件dbgp
- xdebug.remote_handler=dbgp
- ;是否收集变量
- xdebug.collect_vars = On
- ;是否收集返回值
- xdebug.collect_return = On
- ;是否收集参数
- xdebug.collect_params = On
- ;跟踪输出路径
- xdebug.trace_output_dir="c:\xdebug"
- ;是否开启调试内容
- xdebug.profiler_enable=On
- ;调试输出路径
- xdebug.profiler_output_dir="c:\xdebug"
- xdebug.remote_host=localhost
第二步,打开netbeans,在工具-选项-php-调试框内,把调试器端口写成和php.ini内xdebug.remote_port一致,点击应用-确定。
第三步,新建项目,也就是把thinkphp创建的项目导入进去,进入项目内,点击调试项目,之后会弹出一个会话框,在项目URL和索引文件填入正确的信息,之后便可以调试了。
第四步,开始调试。进入项目的一个具体页面,点击调试文件或是调试项目,之后调试便开始了,按F7键进行调试。下面,我便把调试后的的心得写下来。
1) 由于thinkphp是单入口的文件,因此调试界面首先便跳到index.php中,该页面主要负责定义项目名称、项目路径、thinkphp所在路径、是否开启调试模式,之后便引入ThinkPHP.php。
2)在ThinkPHP.php页面里,主要是定义了项目开始运行时间、开始时的内存使用情况、重新检测了是否定义项目路径、缓存目录路径、项目是否开始调试模式、~runtime.php存放地址,若是项目开启调试,则进入THINK_PATH.'Common/runtime.php',没有开启则直接进入已编译好的~runtime.php.,这里属于调试模式,即APP_DEBUG为TRUE。
3)runtime.php文件主要负责检测是否定义thinkphp路径,没有的话就退出,这应该是阻止恶意链接吧,定义版本信息,检测php的版本,定义系统常量、URL的4种模式、目录常量(tp系统的和项目的),加载运行所需文件,记录加载文件文件时间,执行Think::Start()。这里着重解释下load_runtime_file()函数。
-
- function load_runtime_file() {
-
-
-
- require THINK_PATH.'Common/common.php';
-
-
-
-
-
- $list = array(
- CORE_PATH.'Core/Think.class.php',
- CORE_PATH.'Core/ThinkException.class.php',
- CORE_PATH.'Core/Behavior.class.php',
- );
-
-
- foreach ($list as $key=>$file){
-
-
-
-
- if(is_file($file)) require_cache($file);
- }
-
-
-
-
-
- if(!is_dir(LIB_PATH)) {
-
-
- build_app_dir();
- }elseif(!is_dir(CACHE_PATH)){
-
-
- check_runtime();
- }elseif(APP_DEBUG){
-
-
- if(is_file(RUNTIME_FILE)) unlink(RUNTIME_FILE);
- }
- }
4)进入Thinkphp,执行静态方法Start(),该方法重新定义了异常、错误、自动加载机制,执行Think::buildApp()方法
a)Think::buildApp()方法主要实现了thinkphp官方手册中的“惯例配置->项目配置->调试配置->分组配置->扩展配置->动态配置”说明。
-
- C(include THINK_PATH.'Conf/convention.php');
- C(include THINK_PATH.'Conf/config.php');
-
-
- C(include CONF_PATH.'config.php');
-
-
- L(include THINK_PATH.'Lang/'.strtolower(C('DEFAULT_LANG')).'.php');
-
-
- C('extends', include THINK_PATH.'Conf/tags.php');
-
-
- C('tags', include CONF_PATH.'tags.php');
-
- THINK_PATH.'Common/functions.php',
- CORE_PATH.'Core/Log.class.php',
- CORE_PATH.'Core/Dispatcher.class.php',
- CORE_PATH.'Core/App.class.php',
- CORE_PATH.'Core/Action.class.php',
- CORE_PATH.'Core/View.class.php',
-
-
- include COMMON_PATH.'common.php';
-
-
- include THINK_PATH.'Conf/alias.php';
-
-
- include CONF_PATH.'alias.php';
-
-
- C(include THINK_PATH.'Conf/debug.php');
-
-
- C( include CONF_PATH . C('APP_STATUS') . '.php');
5)上述方法执行完后,进入App.class.php中执行App::run()方法。
a)tag('app_init'),项目初始化标签,在tag.php文件中可以看到,app_init对应的值为空,因此这个标签是什么也没执行。之后在第九条详细解释tag方法。
6)执行App::init()方法,该方法首先设置时区,默认为PRC,可以通过在项目中配置DEFAULT_TIMEZONE的值来修改,之后加载动态项目公共文件和配置
a)load_ext_file()方法主要是加载动态扩展文件
-
-
-
-
- function load_ext_file() {
-
-
-
-
-
-
- if (C('LOAD_EXT_FILE')) {
- $files = explode(',', C('LOAD_EXT_FILE'));
- foreach ($files as $file) {
- $file = COMMON_PATH . $file . '.php';
- if (is_file($file))
- include $file;
- }
- }
-
-
-
-
- if (C('LOAD_EXT_CONFIG')) {
- $configs = C('LOAD_EXT_CONFIG');
- if (is_string($configs))
- $configs = explode(',', $configs);
- foreach ($configs as $key => $config) {
- $file = CONF_PATH . $config . '.php';
- if (is_file($file)) {
- is_numeric($key) ? C(include $file) : C($key, include $file);
- }
- }
- }
-
-
-
- if (C('OS_CONFIG')) {
- if (preg_match('/WIN/', PHP_OS)) {
- $file = CONF_PATH . 'win.php';
- } else {
- $file = CONF_PATH . 'linux.php';
- }
- if (is_file($file))
- C(include $file);
- }
- }
7)进入Dispatcher.class.php执行Dispatcher::dispatch()方法。
-
-
-
-
-
- static public function dispatch() {
-
-
- $urlMode = C('URL_MODEL');
-
-
-
-
- if(!empty($_GET[C('VAR_PATHINFO')])) {
- $_SERVER['PATH_INFO'] = $_GET[C('VAR_PATHINFO')];
- unset($_GET[C('VAR_PATHINFO')]);
- }
-
-
- if($urlMode == URL_COMPAT ){
-
-
- define('PHP_FILE',_PHP_FILE_.'?'.C('VAR_PATHINFO').'=');
-
-
-
- }elseif($urlMode == URL_REWRITE ) {
-
-
- $url = dirname(_PHP_FILE_);
- if($url == '/' || $url == '\\')
- $url = '';
- define('PHP_FILE',$url);
-
-
- }else {
-
-
- define('PHP_FILE',_PHP_FILE_);
- }
-
-
-
- if(C('APP_SUB_DOMAIN_DEPLOY')) {
- $rules = C('APP_SUB_DOMAIN_RULES');
- $subDomain = strtolower(substr($_SERVER['HTTP_HOST'],0,strpos($_SERVER['HTTP_HOST'],'.')));
- define('SUB_DOMAIN',$subDomain);
- if($subDomain && isset($rules[$subDomain])) {
- $rule = $rules[$subDomain];
- }elseif(isset($rules['*'])){
- if('www' != $subDomain && !in_array($subDomain,C('APP_SUB_DOMAIN_DENY'))) {
- $rule = $rules['*'];
- }
- }
- if(!empty($rule)) {
-
- $array = explode('/',$rule[0]);
- $module = array_pop($array);
- if(!empty($module)) {
- $_GET[C('VAR_MODULE')] = $module;
- $domainModule = true;
- }
- if(!empty($array)) {
- $_GET[C('VAR_GROUP')] = array_pop($array);
- $domainGroup = true;
- }
- if(isset($rule[1])) {
- parse_str($rule[1],$parms);
- $_GET = array_merge($_GET,$parms);
- }
- }
- }
-
-
-
- if(empty($_SERVER['PATH_INFO'])) {
- $types = explode(',',C('URL_PATHINFO_FETCH'));
- foreach ($types as $type){
- if(0===strpos($type,':')) {
- $_SERVER['PATH_INFO'] = call_user_func(substr($type,1));
- break;
- }elseif(!empty($_SERVER[$type])) {
- $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type],$_SERVER['SCRIPT_NAME']))?
- substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
- break;
- }
- }
- }
-
-
- $depr = C('URL_PATHINFO_DEPR');
-
- if(!empty($_SERVER['PATH_INFO'])) {
-
-
- tag('path_info');
-
-
- $part = pathinfo($_SERVER['PATH_INFO']);
-
-
- define('__EXT__', isset($part['extension'])?strtolower($part['extension']):'');
- if(C('URL_HTML_SUFFIX')) {
- $_SERVER['PATH_INFO'] = preg_replace('/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i', '', $_SERVER['PATH_INFO']);
- }elseif(__EXT__) {
- $_SERVER['PATH_INFO'] = preg_replace('/.'.__EXT__.'$/i','',$_SERVER['PATH_INFO']);
- }
-
-
-
- if(!self::routerCheck()){
-
-
- $paths = explode($depr,trim($_SERVER['PATH_INFO'],'/'));
-
-
- if(C('VAR_URL_PARAMS')) {
-
-
- $_GET[C('VAR_URL_PARAMS')] = $paths;
- }
-
- $var = array();
-
-
-
- if (C('APP_GROUP_LIST') && !isset($_GET[C('VAR_GROUP')])){
- $var[C('VAR_GROUP')] = in_array(strtolower($paths[0]),explode(',',strtolower(C('APP_GROUP_LIST'))))? array_shift($paths) : '';
- if(C('APP_GROUP_DENY') && in_array(strtolower($var[C('VAR_GROUP')]),explode(',',strtolower(C('APP_GROUP_DENY'))))) {
-
- exit;
- }
- }
-
-
- if(!isset($_GET[C('VAR_MODULE')])) {
- $var[C('VAR_MODULE')] = array_shift($paths);
- }
- $var[C('VAR_ACTION')] = array_shift($paths);
-
-
- preg_replace('@(\w+)\/([^\/]+)@e', '$var[\'\\1\']=strip_tags(\'\\2\');', implode('/',$paths));
-
-
- $_GET = array_merge($var,$_GET);
- }
- define('__INFO__',$_SERVER['PATH_INFO']);
- }
-
-
- define('__SELF__',strip_tags($_SERVER['REQUEST_URI']));
-
-
- define('__APP__',strip_tags(PHP_FILE));
-
-
- if (C('APP_GROUP_LIST')) {
-
-
-
- define('GROUP_NAME', self::getGroup(C('VAR_GROUP')));
-
-
- define('__GROUP__',(!empty($domainGroup) || strtolower(GROUP_NAME) == strtolower(C('DEFAULT_GROUP')) )?__APP__ : __APP__.'/'.GROUP_NAME);
- }
-
-
- define('BASE_LIB_PATH', (defined('GROUP_NAME') && C('APP_GROUP_MODE')==1) ? APP_PATH.C('APP_GROUP_PATH').'/'.GROUP_NAME.'/' : LIB_PATH);
- if(defined('GROUP_NAME')) {
- if(1 == C('APP_GROUP_MODE')){
- $config_path = BASE_LIB_PATH.'Conf/';
- $common_path = BASE_LIB_PATH.'Common/';
- }else{
- $config_path = CONF_PATH.GROUP_NAME.'/';
- $common_path = COMMON_PATH.GROUP_NAME.'/';
- }
-
-
- if(is_file($config_path.'config.php'))
- C(include $config_path.'config.php');
-
-
- if(is_file($common_path.'function.php'))
- include $common_path.'function.php';
- }
-
-
- define('MODULE_NAME',self::getModule(C('VAR_MODULE')));
- define('ACTION_NAME',self::getAction(C('VAR_ACTION')));
-
-
- $moduleName = defined('MODULE_ALIAS')?MODULE_ALIAS:MODULE_NAME;
- if(defined('GROUP_NAME')) {
- define('__URL__',!empty($domainModule)?__GROUP__.$depr : __GROUP__.$depr.$moduleName);
- }else{
- define('__URL__',!empty($domainModule)?__APP__.'/' : __APP__.'/'.$moduleName);
- }
- $tmpAppUrl = explode('/', __URL__);
-
-
- define('__APPURL__', './');
-
-
- define('__ACTION__',__URL__.$depr.(defined('ACTION_ALIAS')?ACTION_ALIAS:ACTION_NAME));
-
-
- $_REQUEST = array_merge($_POST,$_GET);
- }
8)回到App::init()方法,定义当前请求的系统常量、URL调度结束、页面压缩输出支持、系统变量安全过滤(过滤$_GET、$_POST)、设置模板相关信息、动态配置异常界面(若是觉得tp异常界面不友好的话,可以重新设计下界面,并在放在项目目录内,在配置文件内通过TMPL_EXCEPTION_FILE来设定)
9)回到App::run()方法,执行tag('app_begin')方法,这里详细介绍下tag方法执行机制
-
-
-
-
-
-
- function tag($tag, &$params=NULL) {
-
-
-
-
-
-
-
- $extends = C('extends.' . $tag);
-
-
- $tags = C('tags.' . $tag);
-
-
- if (!empty($tags)) {
-
-
- if(empty($tags['_overlay']) && !empty($extends)) {
- $tags = array_unique(array_merge($extends,$tags));
-
-
- } elseif (isset($tags['_overlay'])){
- unset($tags['_overlay']);
- }
-
-
- } elseif (!empty($extends)) {
- $tags = $extends;
- }
-
- if($tags) {
-
- if(APP_DEBUG) {
- G($tag.'Start');
- trace('[ '.$tag.' ] --START--','','INFO');
- }
-
-
- foreach ($tags as $key=>$name) {
-
-
- if(!is_int($key)) {
- $name = $key;
- }
-
-
- B($name, $params);
- }
-
-
- if(APP_DEBUG) {
- trace('[ '.$tag.' ] --END-- [ RunTime:'.G($tag.'Start',$tag.'End',6).'s ]','','INFO');
- }
-
-
- } else {
- return false;
- }
- }
-
-
-
-
-
-
-
- function B($name, &$params=NULL) {
-
-
- $class = $name.'Behavior';
- if(APP_DEBUG) {
- G('behaviorStart');
- }
-
-
-
-
-
- $behavior = new $class();
-
-
- $behavior->run($params);
- if(APP_DEBUG) {
- G('behaviorEnd');
- trace('Run '.$name.' Behavior [ RunTime:'.G('behaviorStart','behaviorEnd',6).'s ]','','INFO');
- }
- }
-
-
-
-
-
-
-
- public static function autoload($class) {
-
-
- if(alias_import($class)) return ;
- $libPath = defined('BASE_LIB_PATH')?BASE_LIB_PATH:LIB_PATH;
- $group = defined('GROUP_NAME') && C('APP_GROUP_MODE')==0 ?GROUP_NAME.'/':'';
-
-
- $file = $class.'.class.php';
-
-
-
- if(substr($class,-8)=='Behavior') {
- if(require_array(array(
- CORE_PATH.'Behavior/'.$file,
- EXTEND_PATH.'Behavior/'.$file,
- LIB_PATH.'Behavior/'.$file,
- $libPath.'Behavior/'.$file),true)
- || (defined('MODE_NAME') && require_cache(MODE_PATH.ucwords(MODE_NAME).'/Behavior/'.$file))) {
- return ;
- }
- }elseif(substr($class,-5)=='Model'){
- if(require_array(array(
- LIB_PATH.'Model/'.$group.$file,
- $libPath.'Model/'.$file,
- EXTEND_PATH.'Model/'.$file),true)) {
- return ;
- }
- }elseif(substr($class,-6)=='Action'){
- if(require_array(array(
- LIB_PATH.'Action/'.$group.$file,
- $libPath.'Action/'.$file,
- EXTEND_PATH.'Action/'.$file),true)) {
- return ;
- }
- }elseif(substr($class,0,5)=='Cache'){
- if(require_array(array(
- EXTEND_PATH.'Driver/Cache/'.$file,
- CORE_PATH.'Driver/Cache/'.$file),true)){
- return ;
- }
- }elseif(substr($class,0,2)=='Db'){
- if(require_array(array(
- EXTEND_PATH.'Driver/Db/'.$file,
- CORE_PATH.'Driver/Db/'.$file),true)){
- return ;
- }
- }elseif(substr($class,0,8)=='Template'){
- if(require_array(array(
- EXTEND_PATH.'Driver/Template/'.$file,
- CORE_PATH.'Driver/Template/'.$file),true)){
- return ;
- }
- }elseif(substr($class,0,6)=='TagLib'){
- if(require_array(array(
- EXTEND_PATH.'Driver/TagLib/'.$file,
- CORE_PATH.'Driver/TagLib/'.$file),true)) {
- return ;
- }
- }
-
-
- $paths = explode(',',C('APP_AUTOLOAD_PATH'));
- foreach ($paths as $path){
- if(import($path.'.'.$class))
-
- return ;
- }
- }
接着执行Session初始化操作,记录应用初始化时间
10)前面的准备工作都做完后,接下来便进入执行应用程序App::exec()方法了,到这一步时,基本就是水到渠成的了,利用前面获取的模块和方法通过检测方法是否存在,并通过ReflectionMethod类来获悉当前调试的类的信息,并且判断当前执行方法是否为public属性,不是的话抛出异常,是的话,便调用invoke方法执行类方法,接下里便是自己的代码了,同时注意的是在解析sql语句的时候执行的tp里的方法。
ok,流程跑了一遍,自己感觉清晰多了,以前就是在项目里写代码,从不看tp的源代码的,今天过了下,感觉良好,下次多过几遍,把里面的精髓掌握住,就写到这里。