ThinkPHP内置的Dispatcher类解析

完成URL解析、路由和调度,这是thinkphp内部的注释说明。 闲来无事,对Dispatcher类进行了梳理,感觉Dispatcher类的作用主要有两大作用(自己总结的)

1、定义一些URL解析前后的常量,包括 __SELF__、__MODULE__、__CONTROLLER__、__ACTION__、 MODULE_NAME、CONTROLLER_NAME、ACTION_NAME、PHP_FILE、__APP__等

2、检测模块,并且加载模块所包含的配置文件、函数文件等等(由于本部分代码简单,不再解释)。

由于在定义常量的时候,涉及thinkphp的参数绑定、模块映射,所以主要对这部分进行讲解,方便对thinkphp的参数绑定、模块映射感兴趣的同学查看,其实也很简单。

控制器映射 是通过这个方法实现的,相应的,模块映射、方法映射也是通过相应的方法实现的

static private function getController($var,$urlCase) {
    $controller = (!empty($_GET[$var])? $_GET[$var]:C('DEFAULT_CONTROLLER'));
    unset($_GET[$var]);
    if($maps = C('URL_CONTROLLER_MAP')) {
        if(isset($maps[strtolower($controller)])) {
            // 记录当前别名
            define('CONTROLLER_ALIAS',strtolower($controller));
            // 获取实际的控制器名
            return   ucfirst($maps[CONTROLLER_ALIAS]);
        }elseif(array_search(strtolower($controller),$maps)){
            // 禁止访问原始控制器
            return   '';
        }
    }
    if($urlCase) {
        // URL地址不区分大小写
        // 智能识别方式 user_type 识别到 UserTypeController 控制器
        $controller = parse_name($controller,1);
    }
    return strip_tags(ucfirst($controller));
}

类的源码,里面自己添加些注释,发现这个类很简单。

static public function dispatch() {

    // 兼容模式PATHINFO获取变量例如 ?s=/module/action/id/1 后面的参数取决于URL_PATHINFO_DEPR
    $varPath        =   C('VAR_PATHINFO'); //默认为 s
    // 默认的插件控制器命名空间变量
    $varAddon       =   C('VAR_ADDON');
    // 默认模块获取变量 默认为 m
    $varModule      =   C('VAR_MODULE');
    // 默认控制器获取变量 默认为 c
    $varController  =   C('VAR_CONTROLLER');
    // 默认操作获取变量 默认为 a
    $varAction      =   C('VAR_ACTION');
    $urlCase        =   C('URL_CASE_INSENSITIVE');

    //PATH_INFO是服务器状态中的一个参数,通过$_SERVER['PATH_INFO']可以查看内容。
    // echo $_SERVER['PATH_INFO'];
    //Apache默认支持PATH_INFO功能

    if(isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数
        $_SERVER['PATH_INFO'] = $_GET[$varPath];
        unset($_GET[$varPath]);
    }elseif(IS_CLI){ // CLI模式下 index.php module/controller/action/params/...
        $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
    }

    // 开启子域名部署     暂时用不到,所以略过这部分代码
    if(C('APP_SUB_DOMAIN_DEPLOY')) {
        $rules      = C('APP_SUB_DOMAIN_RULES');
        if(isset($rules[$_SERVER['HTTP_HOST']])) { // 完整域名或者IP配置
            define('APP_DOMAIN',$_SERVER['HTTP_HOST']); // 当前完整域名
            $rule = $rules[APP_DOMAIN];
        }else{
            if(strpos(C('APP_DOMAIN_SUFFIX'),'.')){ // com.cn net.cn 
                $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -3);
            }else{
                $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -2);                    
            }
            if(!empty($domain)) {
                $subDomain = implode('.', $domain);
                define('SUB_DOMAIN',$subDomain); // 当前完整子域名
                $domain2   = array_pop($domain); // 二级域名
                if($domain) { // 存在三级域名
                    $domain3 = array_pop($domain);
                }
                if(isset($rules[$subDomain])) { // 子域名
                    $rule = $rules[$subDomain];
                }elseif(isset($rules['*.' . $domain2]) && !empty($domain3)){ // 泛三级域名
                    $rule = $rules['*.' . $domain2];
                    $panDomain = $domain3;
                }elseif(isset($rules['*']) && !empty($domain2) && 'www' != $domain2 ){ // 泛二级域名
                    $rule      = $rules['*'];
                    $panDomain = $domain2;
                }
            }                
        }

        if(!empty($rule)) {
            // 子域名部署规则 '子域名'=>array('模块名[/控制器名]','var1=a&var2=b');
            if(is_array($rule)){
                list($rule,$vars) = $rule;
            }
            $array      =   explode('/',$rule);
            // 模块绑定
            define('BIND_MODULE',array_shift($array));
            // 控制器绑定         
            if(!empty($array)) {
                $controller  =   array_shift($array);
                if($controller){
                    define('BIND_CONTROLLER',$controller);
                }
            }
            if(isset($vars)) { // 传入参数
                parse_str($vars,$parms);
                if(isset($panDomain)){
                    $pos = array_search('*', $parms);
                    if(false !== $pos) {
                        // 泛域名作为参数
                        $parms[$pos] = $panDomain;
                    }                         
                }                   
                $_GET   =  array_merge($_GET,$parms);
            }
        }
    }


    // 分析PATHINFO信息
    if(!isset($_SERVER['PATH_INFO'])) {
        //http://www.kancloud.cn/manual/thinkphp/1865   PATH_INFO支持
        //'URL_PATHINFO_FETCH'    =>  'ORIG_PATH_INFO,REDIRECT_PATH_INFO,REDIRECT_URL', // 用于兼容判断PATH_INFO 参数的SERVER替代变量列表
        $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;
            }
        }
    }


    // PATHINFO模式下,各参数之间的分割符号
    $depr = C('URL_PATHINFO_DEPR');
    define('MODULE_PATHINFO_DEPR',  $depr);


    if(empty($_SERVER['PATH_INFO'])) {
        $_SERVER['PATH_INFO'] = '';
        define('__INFO__','');
        define('__EXT__','');
    }else{
        define('__INFO__',trim($_SERVER['PATH_INFO'],'/'));
        // URL后缀
        define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'],PATHINFO_EXTENSION)));
        $_SERVER['PATH_INFO'] = __INFO__;     
        if(!defined('BIND_MODULE') && (!C('URL_ROUTER_ON') || !Route::check())){
            if (__INFO__ && C('MULTI_MODULE')){ // 获取模块名
                $paths      =   explode($depr,__INFO__,2);
                $allowList  =   C('MODULE_ALLOW_LIST'); // 允许的模块列表
                $module     =   preg_replace('/\.' . __EXT__ . '$/i', '',$paths[0]);
                if( empty($allowList) || (is_array($allowList) && in_array_case($module, $allowList))){
                    $_GET[$varModule]       =   $module;
                    $_SERVER['PATH_INFO']   =   isset($paths[1])?$paths[1]:'';
                }
            }
        }             
    }

    // URL常量
    define('__SELF__',strip_tags($_SERVER[C('URL_REQUEST_URI')]));



    // 获取模块名称
    define('MODULE_NAME', defined('BIND_MODULE')? BIND_MODULE : self::getModule($varModule));
    
    // 检测模块是否存在
    if( MODULE_NAME && (defined('BIND_MODULE') || !in_array_case(MODULE_NAME,C('MODULE_DENY_LIST')) ) && is_dir(APP_PATH.MODULE_NAME)){
        // 定义当前模块路径
        define('MODULE_PATH', APP_PATH.MODULE_NAME.'/');
        // 定义当前模块的模版缓存路径
        C('CACHE_PATH',CACHE_PATH.MODULE_NAME.'/');
        // 定义当前模块的日志目录
     C('LOG_PATH',  realpath(LOG_PATH).'/'.MODULE_NAME.'/');

        // 模块检测
        Hook::listen('module_check');

        // 加载模块配置文件
        if(is_file(MODULE_PATH.'Conf/config'.CONF_EXT))
            C(load_config(MODULE_PATH.'Conf/config'.CONF_EXT));
        // 加载应用模式对应的配置文件
        if('common' != APP_MODE && is_file(MODULE_PATH.'Conf/config_'.APP_MODE.CONF_EXT))
            C(load_config(MODULE_PATH.'Conf/config_'.APP_MODE.CONF_EXT));
        // 当前应用状态对应的配置文件
        if(APP_STATUS && is_file(MODULE_PATH.'Conf/'.APP_STATUS.CONF_EXT))
            C(load_config(MODULE_PATH.'Conf/'.APP_STATUS.CONF_EXT));

        // 加载模块别名定义
        if(is_file(MODULE_PATH.'Conf/alias.php'))
            Think::addMap(include MODULE_PATH.'Conf/alias.php');
        // 加载模块tags文件定义
        if(is_file(MODULE_PATH.'Conf/tags.php'))
            Hook::import(include MODULE_PATH.'Conf/tags.php');
        // 加载模块函数文件
        if(is_file(MODULE_PATH.'Common/function.php'))
            include MODULE_PATH.'Common/function.php';
        
        $urlCase        =   C('URL_CASE_INSENSITIVE');
        // 加载模块的扩展配置文件
        load_ext_file(MODULE_PATH);
    }else{
        E(L('_MODULE_NOT_EXIST_').':'.MODULE_NAME);
    }

    if(!defined('__APP__')){
     $urlMode        =   C('URL_MODEL');
        //定义三种模式下的 PHP_FILE
        //  一般打印出的结果为 /sae/1/StudyRich/index.php
     if($urlMode == URL_COMPAT ){// 兼容模式判断
         define('PHP_FILE',_PHP_FILE_.'?'.$varPath.'=');
     }elseif($urlMode == URL_REWRITE ) {
         $url    =   dirname(_PHP_FILE_);
         if($url == '/' || $url == '\\')
             $url    =   '';
         define('PHP_FILE',$url);
     }else {
         define('PHP_FILE',_PHP_FILE_);
     }
     // 当前应用地址
     define('__APP__',strip_tags(PHP_FILE));
 }

    // 模块URL地址
    $moduleName    =   defined('MODULE_ALIAS')? MODULE_ALIAS : MODULE_NAME;
    define('__MODULE__',(defined('BIND_MODULE') || !C('MULTI_MODULE'))? __APP__ : __APP__.'/'.($urlCase ? strtolower($moduleName) : $moduleName));

    if('' != $_SERVER['PATH_INFO'] && (!C('URL_ROUTER_ON') ||  !Route::check()) ){   // 检测路由规则 如果没有则按默认规则调度URL
        Hook::listen('path_info');
        // 检查禁止访问的URL后缀
        if(C('URL_DENY_SUFFIX') && preg_match('/\.('.trim(C('URL_DENY_SUFFIX'),'.').')$/i', $_SERVER['PATH_INFO'])){
            send_http_status(404);
            exit;
        }
        
        // 去除URL后缀
        $_SERVER['PATH_INFO'] = preg_replace(C('URL_HTML_SUFFIX')? '/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i' : '/\.'.__EXT__.'$/i', '', $_SERVER['PATH_INFO']);

        $depr   =   C('URL_PATHINFO_DEPR');
        //此时$_SERVER['PATH_INFO'] 已经不包含 模块名
        // index/login/status/2
        $paths  =   explode($depr,trim($_SERVER['PATH_INFO'],$depr));

        if(!defined('BIND_CONTROLLER')) {// 获取控制器
            if(C('CONTROLLER_LEVEL')>1){// 控制器层次
                $_GET[$varController]   =   implode('/',array_slice($paths,0,C('CONTROLLER_LEVEL')));
                $paths  =   array_slice($paths, C('CONTROLLER_LEVEL'));
            }else{
                //array_shift 删除数组第一个元素,并返回该元素
                $_GET[$varController]   =   array_shift($paths);
            }
        }
        // 获取操作
        if(!defined('BIND_ACTION')){
            $_GET[$varAction]  =   array_shift($paths);
        }
        // 解析剩余的URL参数
        $var  =  array();

        if(C('URL_PARAMS_BIND') && 1 == C('URL_PARAMS_BIND_TYPE')){
            // URL参数按顺序绑定变量
            $var    =   $paths;
        }else{
            // URL变量绑定的类型 0 按变量名绑定 1 按变量顺序绑定
            preg_replace_callback('/(\w+)\/([^\/]+)/', function($match) use(&$var){
                $var[$match[1]]=strip_tags($match[2]);
            }, implode('/',$paths));
        }
        $_GET   =  array_merge($var,$_GET);
    }
    // 获取控制器的命名空间(路径)
    define('CONTROLLER_PATH',   self::getSpace($varAddon,$urlCase));
    // 获取控制器和操作名
    define('CONTROLLER_NAME',   defined('BIND_CONTROLLER')? BIND_CONTROLLER : self::getController($varController,$urlCase));
    define('ACTION_NAME',       defined('BIND_ACTION')? BIND_ACTION : self::getAction($varAction,$urlCase));


    // 当前控制器的UR地址
    $controllerName    =   defined('CONTROLLER_ALIAS')? CONTROLLER_ALIAS : CONTROLLER_NAME;
    define('__CONTROLLER__',__MODULE__.$depr.(defined('BIND_CONTROLLER')? '': ( $urlCase ? parse_name($controllerName) : $controllerName )) );

    // 当前操作的URL地址
    define('__ACTION__',__CONTROLLER__.$depr.(defined('ACTION_ALIAS')?ACTION_ALIAS:ACTION_NAME));

    //保证$_REQUEST正常取值
    $_REQUEST = array_merge($_POST,$_GET,$_COOKIE);    // -- 加了$_COOKIE.  保证哦..
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值