讓Thinkphp 3.1.x支持php7以上的語法

字號+ 編輯: 种花家 修訂: 种花家 來源: 原创 2023-11-15 我要說兩句(3)

讓tp3支持php7? 支持php8? 太冒險了吧? 但是如果必須改呢? 你的項目希望總是用php5.4嗎? ok本文告訴你怎麽玩

很多程序員以爲現在時興的golang才是王者(2023更新: 今年基本涼了)。擺一個殘酷的現實告訴你,ThinkPHP3.x系列框架因太容易使用, 在2022年國内中小企業,甚至是某些大型國企商業開發應用的出場概率大到你不敢想象。好多二線三線城市的小廠還在用它搞開發。不信?那你接到的項目一定很高耑嘍!筆者這邊好多大佬頭疼於TP的老項目, 但他們沒有想重構成golang或者rust。最起碼把php升級一下版本好伐!OK問題來了, 遷移了項目之後, 十有八九, 前耑顯示是空白的。何解?

於是有的人折騰index.php, 加入各種debug專用配置, 妄圖找到錯誤的點在哪裡, 然後逐個擊破:

ini_set('display_errors', '1');
error_reporting(E_ALL);

define("APP_DEBUG", true);
define('APP_ERROR_HANDLE',false);
define("SHOW_ERROR", true);

然並卵。排除代碼質量的問題先不說,根本性的問題都來源於框架的一個過時的模板正則語法解析類, 解決了就ok了, 但是改那個正則又很惡心。

解決方案

找到框架裡一個文档名叫 ThinkTemplate.class.php 的模板正則處理類。把以下代碼拷進去。注意,改的其實都是preg_match相關的語句, 你可以搜索preg_match關鍵字試試。

<?php
 
/**
 * ThinkPHP内置模板引擎類
 * 支持XML標簽和普通標簽的模板解析
 * 編譯型模板引擎 支持動態緩存
 * @category   Think
 * @package  Think
 * @subpackage  Template
 * @author liu21st <liu21st@gmail.com>
 */
class ThinkTemplate
{
 
    // 模板頁面中引入的標簽庫列表
    protected   $tagLib          =   array();
    // 當前模板文档
    protected   $templateFile    =   '';
    // 模板變量
    public      $tVar            =   array();
    public      $config          =   array();
    private     $literal         =   array();
    private     $block           =   array();
 
    /**
     * 架構函數
     * @access public
     */
    public function __construct()
    {
        $this->config['cache_path']         =   C('CACHE_PATH');
        $this->config['template_suffix']    =   C('TMPL_TEMPLATE_SUFFIX');
        $this->config['cache_suffix']       =   C('TMPL_CACHFILE_SUFFIX');
        $this->config['tmpl_cache']         =   C('TMPL_CACHE_ON');
        $this->config['cache_time']         =   C('TMPL_CACHE_TIME');
        $this->config['taglib_begin']       =   $this->stripPreg(C('TAGLIB_BEGIN'));
        $this->config['taglib_end']         =   $this->stripPreg(C('TAGLIB_END'));
        $this->config['tmpl_begin']         =   $this->stripPreg(C('TMPL_L_DELIM'));
        $this->config['tmpl_end']           =   $this->stripPreg(C('TMPL_R_DELIM'));
        $this->config['default_tmpl']       =   C('TEMPLATE_NAME');
        $this->config['layout_item']        =   C('TMPL_LAYOUT_ITEM');
    }
 
    private function stripPreg($str)
    {
        return str_replace(
            array('{', '}', '(', ')', '|', '[', ']', '-', '+', '*', '.', '^', '?'),
            array('\{', '\}', '\(', '\)', '\|', '\[', '\]', '\-', '\+', '\*', '\.', '\^', '\?'),
            $str
        );
    }
 
    // 模板變量獲取和設置
    public function get($name)
    {
        if (isset($this->tVar[$name]))
            return $this->tVar[$name];
        else
            return false;
    }
 
    public function set($name, $value)
    {
        $this->tVar[$name] = $value;
    }
 
    /**
     * 加載模板
     * @access public
     * @param string $tmplTemplateFile 模板文档
     * @param array  $templateVar 模板變量
     * @param string $prefix 模板標識前綴
     * @return void
     */
    public function fetch($templateFile, $templateVar, $prefix = '')
    {
        $this->tVar         =   $templateVar;
        $templateCacheFile  =   $this->loadTemplate($templateFile, $prefix);
        // 模板陣列變量分解成爲獨立變量
        extract($templateVar, EXTR_OVERWRITE);
        //載入模版緩存文档
        include $templateCacheFile;
    }
 
    /**
     * 加載主模板並緩存
     * @access public
     * @param string $tmplTemplateFile 模板文档
     * @param string $prefix 模板標識前綴
     * @return string
     * @throws ThinkExecption
     */
    public function loadTemplate($tmplTemplateFile, $prefix = '')
    {
        if (is_file($tmplTemplateFile)) {
            $this->templateFile    =  $tmplTemplateFile;
            // 讀取模板文档内容
            $tmplContent =  file_get_contents($tmplTemplateFile);
        } else {
            $tmplContent =  $tmplTemplateFile;
        }
        // 根據模版文档名定位緩存文档
        $tmplCacheFile = $this->config['cache_path'] . $prefix . md5($tmplTemplateFile) . $this->config['cache_suffix'];
 
        // 判斷是否啓用布局
        if (C('LAYOUT_ON')) {
            if (false !== strpos($tmplContent, '{__NOLAYOUT__}')) { // 可以單獨定義不使用布局
                $tmplContent = str_replace('{__NOLAYOUT__}', '', $tmplContent);
            } else { // 替換布局的主體内容
                $layoutFile  =  THEME_PATH . C('LAYOUT_NAME') . $this->config['template_suffix'];
                $tmplContent = str_replace($this->config['layout_item'], $tmplContent, file_get_contents($layoutFile));
            }
        }
        // 編譯模板内容
        $tmplContent =  $this->compiler($tmplContent);
        // 檢測模板目錄
        $dir         =  dirname($tmplCacheFile);
        if (!is_dir($dir))
            mkdir($dir, 0755, true);
        //重寫Cache文档
        if (false === file_put_contents($tmplCacheFile, trim($tmplContent)))
            throw_exception(L('_CACHE_WRITE_ERROR_') . ':' . $tmplCacheFile);
        return $tmplCacheFile;
    }
 
    /**
     * 編譯模板文档内容
     * @access protected
     * @param mixed $tmplContent 模板内容
     * @return string
     */
    protected function compiler($tmplContent)
    {
        // 模板解析
        $tmplContent =  $this->parse($tmplContent);
        // 還原被替換的Literal標簽
        $tmplContent =  preg_replace_callback('/<!--###literal(\d+)###-->/is', function ($match) {
            return $this->restoreLiteral($match[1]);
        }, $tmplContent);
        // 添加安全代碼
        $tmplContent =  '<?php if (!defined(\'THINK_PATH\')) exit();?>' . $tmplContent;
        if (C('TMPL_STRIP_SPACE')) {
            /* 去除html空格與換行 */
            $find           = array('~>\s+<~', '~>(\s+\n|\r)~');
            $replace        = array('><', '>');
            $tmplContent    = preg_replace($find, $replace, $tmplContent);
        }
        // 優化生成的php代碼
        $tmplContent = str_replace('?><?php', '', $tmplContent);
        return strip_whitespace($tmplContent);
    }
 
    /**
     * 模板解析入口
     * 支持普通標簽和TagLib解析 支持自定義標簽庫
     * @access public
     * @param string $content 要解析的模板内容
     * @return string
     */
    public function parse($content)
    {
        // 内容爲空不解析
        if (empty($content)) return '';
        $begin      =   $this->config['taglib_begin'];
        $end        =   $this->config['taglib_end'];
 
        // 檢查include語法
        $content    =   $this->parseInclude($content);
 
        // 檢查PHP語法
        $content    =   $this->parsePhp($content);
        // 首先替換literal標簽内容
        $content = preg_replace_callback('/' . $begin . 'literal' . $end . '(.*?)' . $begin . '\/literal' . $end . '/is', function ($match) {
            $this->parseLiteral($match[1]);
        }, $content);
 
        // 獲取需要引入的標簽庫列表
        // 標簽庫只需要定義一次,允許引入多個一次
        // 一般放在文档的最前面
        // 格式:<taglib name="html,mytag..." />
        // 當TAGLIB_LOAD配置爲true時才會進行檢測
        if (C('TAGLIB_LOAD')) {
            $this->getIncludeTagLib($content);
            if (!empty($this->tagLib)) {
                // 對導入的TagLib進行解析
                foreach ($this->tagLib as $tagLibName) {
                    $this->parseTagLib($tagLibName, $content);
                }
            }
        }
        // 預先加載的標簽庫 無需在每個模板中使用taglib標簽加載 但必須使用標簽庫XML前綴
        if (C('TAGLIB_PRE_LOAD')) {
            $tagLibs =  explode(',', C('TAGLIB_PRE_LOAD'));
            foreach ($tagLibs as $tag) {
                $this->parseTagLib($tag, $content);
            }
        }
        // 内置標簽庫 無需使用taglib標簽導入就可以使用 並且不需使用標簽庫XML前綴
        $tagLibs =  explode(',', C('TAGLIB_BUILD_IN'));
        foreach ($tagLibs as $tag) {
            $this->parseTagLib($tag, $content, true);
        }
        // 解析普通模板標簽 {tagName}
        return preg_replace_callback('/(' . $this->config['tmpl_begin'] . ')([^\d\s' . $this->config['tmpl_begin'] . $this->config['tmpl_end'] . '].+?)(' . $this->config['tmpl_end'] . ')/is', function ($match) {
            return $this->parseTag($match[2]);
        }, $content);
    }
 
    // 檢查PHP語法
    protected function parsePhp($content)
    {
        if (ini_get('short_open_tag')) {
            // 開啓短標簽的情況要將<?標簽用echo方式輸出 否則無法正常輸出xml標識
            $content = preg_replace('/(<\?(?!php|=|$))/i', '<?php echo \'\\1\'; ?>' . "\n", $content);
        }
        // PHP語法檢查
        if (C('TMPL_DENY_PHP') && false !== strpos($content, '<?php')) {
            throw_exception(L('_NOT_ALLOW_PHP_'));
        }
        return $content;
    }
 
    // 解析模板中的布局標簽
    protected function parseLayout($content)
    {
        // 讀取模板中的布局標簽
        $find = preg_match('/' . $this->config['taglib_begin'] . 'layout\s(.+?)\s*?\/' . $this->config['taglib_end'] . '/is', $content, $matches);
        if ($find) {
            // 替換Layout標簽
            $content    =   str_replace($matches[0], '', $content);
            // 解析Layout標簽
            $array      =   $this->parseXmlAttrs($matches[1]);
            if (!C('LAYOUT_ON') || C('LAYOUT_NAME') != $array['name']) {
                // 讀取布局模板
                $layoutFile =   THEME_PATH . $array['name'] . $this->config['template_suffix'];
                $replace    =   isset($array['replace']) ? $array['replace'] : $this->config['layout_item'];
                // 替換布局的主體内容
                $content    =   str_replace($replace, $content, file_get_contents($layoutFile));
            }
        } else {
            $content = str_replace('{__NOLAYOUT__}', '', $content);
        }
        return $content;
    }
 
    // 解析模板中的include標簽
    protected function parseInclude($content)
    {
 
        // 解析繼承
        $content    =   $this->parseExtend($content);
 
        // 解析布局
        $content    =   $this->parseLayout($content);
        // 讀取模板中的include標簽
        $find       =   preg_match_all('/' . $this->config['taglib_begin'] . 'include\s(.+?)\s*?\/' . $this->config['taglib_end'] . '/is', $content, $matches);
 
        if ($find) {
            for ($i = 0; $i < $find; $i++) {
                $include    =   $matches[1][$i];
                $array      =   $this->parseXmlAttrs($include);
                $file       =   $array['file'];
                unset($array['file']);
                $content    =   str_replace($matches[0][$i], $this->parseIncludeItem($file, $array), $content);
            }
        }
        return $content;
    }
 
    // 解析模板中的extend標簽
    protected function parseExtend($content)
    {
        $begin      =   $this->config['taglib_begin'];
        $end        =   $this->config['taglib_end'];
        // 讀取模板中的繼承標簽
        $find       =   preg_match('/' . $begin . 'extend\s(.+?)\s*?\/' . $end . '/is', $content, $matches);
 
        if ($find) {
            // 替換extend標簽
            $content    =   str_replace($matches[0], '', $content);
            // 記錄頁面中的block標簽
            preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
                return $this->parseBlock($match[1], $match[2]);
            }, $content);
            // 讀取繼承模板
            $array      =   $this->parseXmlAttrs($matches[1]);
            $content    =   $this->parseTemplateName($array['name']);
            // 替換block標簽
            return preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
                return $this->replaceBlock($match[1], $match[2]);
            }, $content);
        } else {
            return preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
                return stripslashes($match[1]);
            }, $content);
        }
    }
 
    /**
     * 分析XML屬性
     * @access private
     * @param string $attrs  XML屬性字符串
     * @return array
     */
    private function parseXmlAttrs($attrs)
    {
        $xml        =   '<tpl><tag ' . $attrs . ' /></tpl>';
        $xml        =   simplexml_load_string($xml);
        if (!$xml)
            throw_exception(L('_XML_TAG_ERROR_'));
        $xml        =   (array)($xml->tag->attributes());
        $array      =   array_change_key_case($xml['@attributes']);
        return $array;
    }
 
    /**
     * 替換頁面中的literal標簽
     * @access private
     * @param string $content  模板内容
     * @return string|false
     */
    private function parseLiteral($content)
    {
        if (trim($content) == '')  return '';
        $content            =   stripslashes($content);
        $i                  =   count($this->literal);
        $parseStr           =   "<!--###literal{$i}###-->";
        $this->literal[$i]  =   $content;
        return $parseStr;
    }
 
    /**
     * 還原被替換的literal標簽
     * @access private
     * @param string $tag  literal標簽序號
     * @return string|false
     */
    private function restoreLiteral($tag)
    {
        // 還原literal標簽
        $parseStr   =  $this->literal[$tag];
        // 銷毀literal記錄
        unset($this->literal[$tag]);
        return $parseStr;
    }
 
    /**
     * 記錄當前頁面中的block標簽
     * @access private
     * @param string $name block名稱
     * @param string $content  模板内容
     * @return string
     */
    private function parseBlock($name, $content)
    {
        $this->block[$name]  =   $content;
        return '';
    }
 
    /**
     * 替換繼承模板中的block標簽
     * @access private
     * @param string $name  block名稱
     * @param string $content  模板内容
     * @return string
     */
    private function replaceBlock($name, $content)
    {
        // 替換block標簽 沒有重新定義則使用原來的
        $replace   =  isset($this->block[$name]) ?   $this->block[$name]   :   $content;
        return stripslashes($replace);
    }
 
    /**
     * 搜索模板頁面中包含的TagLib庫
     * 並返回列表
     * @access public
     * @param string $content  模板内容
     * @return string|false
     */
    public function getIncludeTagLib(&$content)
    {
        // 搜索是否有TagLib標簽
        $find = preg_match('/' . $this->config['taglib_begin'] . 'taglib\s(.+?)(\s*?)\/' . $this->config['taglib_end'] . '\W/is', $content, $matches);
        if ($find) {
            // 替換TagLib標簽
            $content        = str_replace($matches[0], '', $content);
            // 解析TagLib標簽
            $array          =   $this->parseXmlAttrs($matches[1]);
            $this->tagLib   = explode(',', $array['name']);
        }
        return;
    }
 
    /**
     * TagLib庫解析
     * @access public
     * @param string $tagLib 要解析的標簽庫
     * @param string $content 要解析的模板内容
     * @param boolen $hide 是否隱藏標簽庫前綴
     * @return string
     */
    public function parseTagLib($tagLib, &$content, $hide = false)
    {
        $begin      =   $this->config['taglib_begin'];
        $end        =   $this->config['taglib_end'];
        $className  =   'TagLib' . ucwords($tagLib);
        $tLib       =   Think::instance($className);
        foreach ($tLib->getTags() as $name => $val) {
            $tags = array($name);
            if (isset($val['alias'])) { // 別名設置
                $tags       = explode(',', $val['alias']);
                $tags[]     =  $name;
            }
            $level      =   isset($val['level']) ? $val['level'] : 1;
            $closeTag   =   isset($val['close']) ? $val['close'] : true;
            foreach ($tags as $tag) {
                $parseTag = !$hide ? $tagLib . ':' . $tag : $tag; // 實際要解析的標簽名稱
                if (!method_exists($tLib, '_' . $tag)) {
                    // 別名可以無需定義解析方法
                    $tag  =  $name;
                }
                $n1 = empty($val['attr']) ? '(\s*?)' : '\s([^' . $end . ']*)';
                if (!$closeTag) {
                    $patterns       = '/' . $begin . $parseTag . $n1 . '\/(\s*?)' . $end . '/is';
                    $content        = preg_replace_callback($patterns, function ($match) use ($tagLib, $tag) {
                        return $this->parseXmlTag($tagLib, $tag, $match[1], '');
                    }, $content);
                } else {
                    $patterns       = '/' . $begin . $parseTag . $n1 . $end . '(.*?)' . $begin . '\/' . $parseTag . '(\s*?)' . $end . '/is';
                    for ($i = 0; $i < $level; $i++) {
                        $content = preg_replace_callback($patterns, function ($match) use ($tagLib, $tag) {
                            return $this->parseXmlTag($tagLib, $tag, $match[1], $match[2]);
                        }, $content);
                    }
                }
            }
        }
    }
 
    /**
     * 解析標簽庫的標簽
     * 需要調用對應的標簽庫文档解析類
     * @access public
     * @param string $tagLib  標簽庫名稱
     * @param string $tag  標簽名
     * @param string $attr  標簽屬性
     * @param string $content  標簽内容
     * @return string|false
     */
    public function parseXmlTag($tagLib, $tag, $attr, $content)
    {
        // if (MAGIC_QUOTES_GPC) {
        $attr   = stripslashes($attr);
        $content = stripslashes($content);
        // }
        if (ini_get('magic_quotes_sybase'))
            $attr   =  str_replace('\"', '\'', $attr);
        $tLib       =  Think::instance('TagLib' . ucwords(strtolower($tagLib)));
        $parse      = '_' . $tag;
        $content    = trim($content);
        return $tLib->$parse($attr, $content);
    }
 
    /**
     * 模板標簽解析
     * 格式: {TagName:args [|content] }
     * @access public
     * @param string $tagStr 標簽内容
     * @return string
     */
    public function parseTag($tagStr)
    {
        // if (MAGIC_QUOTES_GPC) {
        $tagStr = stripslashes($tagStr);
        // }
        // 還原非模板標簽
        if (preg_match('/^[\s|\d]/is', $tagStr))
            // 過濾空格和數字打頭的標簽
            return C('TMPL_L_DELIM') . $tagStr . C('TMPL_R_DELIM');
        $flag   =  substr($tagStr, 0, 1);
        $flag2  =  substr($tagStr, 1, 1);
        $name   = substr($tagStr, 1);
        if ('$' == $flag && '.' != $flag2 && '(' != $flag2) { //解析模板變量 格式 {$varName}
            return $this->parseVar($name);
        } elseif ('-' == $flag || '+' == $flag) { // 輸出計算
            return  '<?php echo ' . $flag . $name . ';?>';
        } elseif (':' == $flag) { // 輸出某個函數的結果
            return  '<?php echo ' . $name . ';?>';
        } elseif ('~' == $flag) { // 執行某個函數
            return  '<?php ' . $name . ';?>';
        } elseif (substr($tagStr, 0, 2) == '//' || (substr($tagStr, 0, 2) == '/*' && substr($tagStr, -2) == '*/')) {
            // 注釋標簽
            return '';
        }
        // 未識別的標簽直接返回
        return C('TMPL_L_DELIM') . $tagStr . C('TMPL_R_DELIM');
    }
 
    /**
     * 模板變量解析,支持使用函數
     * 格式: {$varname|function1|function2=arg1,arg2}
     * @access public
     * @param string $varStr 變量數據
     * @return string
     */
    public function parseVar($varStr)
    {
        $varStr     =   trim($varStr);
        static $_varParseList = array();
        //如果已經解析過該變量字串,則直接返回變量值
        if (isset($_varParseList[$varStr])) return $_varParseList[$varStr];
        $parseStr   =   '';
        $varExists  =   true;
        if (!empty($varStr)) {
            $varArray = explode('|', $varStr);
            //取得變量名稱
            $var = array_shift($varArray);
            if ('Think.' == substr($var, 0, 6)) {
                // 所有以Think.打頭的以特殊變量對待 無需模板賦值就可以輸出
                $name = $this->parseThinkVar($var);
            } elseif (false !== strpos($var, '.')) {
                //支持 {$var.property}
                $vars = explode('.', $var);
                $var  =  array_shift($vars);
                switch (strtolower(C('TMPL_VAR_IDENTIFY'))) {
                    case 'array': // 識別爲數組
                        $name = '$' . $var;
                        foreach ($vars as $key => $val)
                            $name .= '["' . $val . '"]';
                        break;
                    case 'obj':  // 識別爲對象
                        $name = '$' . $var;
                        foreach ($vars as $key => $val)
                            $name .= '->' . $val;
                        break;
                    default:  // 自動判斷數組或對象 只支持二維
                        $name = 'is_array($' . $var . ')?$' . $var . '["' . $vars[0] . '"]:$' . $var . '->' . $vars[0];
                }
            } elseif (false !== strpos($var, '[')) {
                // 支持 {$var['key']} 方式輸出數組
                $name = "$" . $var;
                preg_match('/(.+?)\[(.+?)\]/is', $var, $match);
                $var = $match[1];
            } elseif (false !== strpos($var, ':') && false === strpos($var, '(') && false === strpos($var, '::') && false === strpos($var, '?')) {
                // 支持 {$var:property} 方式輸出對象的屬性
                $vars = explode(':', $var);
                $var  =  str_replace(':', '->', $var);
                $name = "$" . $var;
                $var  = $vars[0];
            } else {
                $name = "$$var";
            }
            //對變量使用函數
            if (count($varArray) > 0)
                $name = $this->parseVarFunction($name, $varArray);
            $parseStr = '<?php echo (' . $name . '); ?>';
        }
        $_varParseList[$varStr] = $parseStr;
        return $parseStr;
    }
 
    /**
     * 對模板變量使用函數
     * 格式 {$varname|function1|function2=arg1,arg2}
     * @access public
     * @param string $name 變量名
     * @param array $varArray  函數列表
     * @return string
     */
    public function parseVarFunction($name, $varArray)
    {
        // 對變量使用函數
        $length = count($varArray);
        // 取得模板禁止使用函數列表
        $template_deny_funs = explode(',', C('TMPL_DENY_FUNC_LIST'));
        for ($i = 0; $i < $length; $i++) {
            $args = explode('=', $varArray[$i], 2);
            // 模板函數過濾
            $fun = strtolower(trim($args[0]));
            switch ($fun) {
                case 'default':  // 特殊模板函數
                    $name   = '(' . $name . ')?(' . $name . '):' . $args[1];
                    break;
                default:  // 通用模板函數
                    if (!in_array($fun, $template_deny_funs)) {
                        if (isset($args[1])) {
                            if (strstr($args[1], '###')) {
                                $args[1] = str_replace('###', $name, $args[1]);
                                $name = "$fun($args[1])";
                            } else {
                                $name = "$fun($name,$args[1])";
                            }
                        } else if (!empty($args[0])) {
                            $name = "$fun($name)";
                        }
                    }
            }
        }
        return $name;
    }
 
    /**
     * 特殊模板變量解析
     * 格式 以 $Think. 打頭的變量屬於特殊模板變量
     * @access public
     * @param string $varStr  變量字符串
     * @return string
     */
    public function parseThinkVar($varStr)
    {
        $vars = explode('.', $varStr);
        $vars[1] = strtoupper(trim($vars[1]));
        $parseStr = '';
        if (count($vars) >= 3) {
            $vars[2] = trim($vars[2]);
            switch ($vars[1]) {
                case 'SERVER':
                    $parseStr = '$_SERVER[\'' . strtoupper($vars[2]) . '\']';
                    break;
                case 'GET':
                    $parseStr = '$_GET[\'' . $vars[2] . '\']';
                    break;
                case 'POST':
                    $parseStr = '$_POST[\'' . $vars[2] . '\']';
                    break;
                case 'COOKIE':
                    if (isset($vars[3])) {
                        $parseStr = '$_COOKIE[\'' . $vars[2] . '\'][\'' . $vars[3] . '\']';
                    } else {
                        $parseStr = 'cookie(\'' . $vars[2] . '\')';
                    }
                    break;
                case 'SESSION':
                    if (isset($vars[3])) {
                        $parseStr = '$_SESSION[\'' . $vars[2] . '\'][\'' . $vars[3] . '\']';
                    } else {
                        $parseStr = 'session(\'' . $vars[2] . '\')';
                    }
                    break;
                case 'ENV':
                    $parseStr = '$_ENV[\'' . strtoupper($vars[2]) . '\']';
                    break;
                case 'REQUEST':
                    $parseStr = '$_REQUEST[\'' . $vars[2] . '\']';
                    break;
                case 'CONST':
                    $parseStr = strtoupper($vars[2]);
                    break;
                case 'LANG':
                    $parseStr = 'L("' . $vars[2] . '")';
                    break;
                case 'CONFIG':
                    if (isset($vars[3])) {
                        $vars[2] .= '.' . $vars[3];
                    }
                    $parseStr = 'C("' . $vars[2] . '")';
                    break;
                default:
                    break;
            }
        } else if (count($vars) == 2) {
            switch ($vars[1]) {
                case 'NOW':
                    $parseStr = "date('Y-m-d g:i a',time())";
                    break;
                case 'VERSION':
                    $parseStr = 'THINK_VERSION';
                    break;
                case 'TEMPLATE':
                    $parseStr = "'" . $this->templateFile . "'"; //'C("TEMPLATE_NAME")';
                    break;
                case 'LDELIM':
                    $parseStr = 'C("TMPL_L_DELIM")';
                    break;
                case 'RDELIM':
                    $parseStr = 'C("TMPL_R_DELIM")';
                    break;
                default:
                    if (defined($vars[1]))
                        $parseStr = $vars[1];
            }
        }
        return $parseStr;
    }
 
    /**
     * 加載公共模板並緩存 和當前模板在同一路徑,否則使用相對路徑
     * @access private
     * @param string $tmplPublicName  公共模板文档名
     * @param array $vars  要傳遞的變量列表
     * @return string
     */
    private function parseIncludeItem($tmplPublicName, $vars = array())
    {
        // 分析模板文档名並讀取内容
        $parseStr = $this->parseTemplateName($tmplPublicName);
        // 替換變量
        foreach ($vars as $key => $val) {
            $parseStr = str_replace('[' . $key . ']', $val, $parseStr);
        }
        // 再次對包含文档進行模板分析
        return $this->parseInclude($parseStr);
    }
 
    /**
     * 分析加載的模板文档並讀取内容 支持多個模板文档讀取
     * @access private
     * @param string $tmplPublicName  模板文档名
     * @return string
     */
    private function parseTemplateName($templateName)
    {
        if (substr($templateName, 0, 1) == '$')
            // 支持加載變量文档名
            $templateName = $this->get(substr($templateName, 1));
        $array  =   explode(',', $templateName);
        $parseStr   =   '';
        foreach ($array as $templateName) {
            if (false === strpos($templateName, $this->config['template_suffix'])) {
                // 解析槼則爲 模板主題:模塊:操作 不支持 跨項目和跨分組調用
                $path   =  explode(':', $templateName);
                $action = array_pop($path);
                $module = !empty($path) ? array_pop($path) : MODULE_NAME;
                if (!empty($path) && THEME_NAME) { // 設置模板主題
                    $path = dirname(THEME_PATH) . '/' . array_pop($path) . '/';
                } else {
                    $path = THEME_PATH;
                }
                $templateName  =  $path . $module . C('TMPL_FILE_DEPR') . $action . $this->config['template_suffix'];
            }
            // 獲取模板文档内容
            $parseStr .= file_get_contents($templateName);
        }
        return $parseStr;
    }
}

 至於其他問題, 你還有改不了的嗎? 請留言。

閲完此文,您的感想如何?
  • 有用

    630

  • 沒用

    20

  • 開心

    122

  • 憤怒

    5

  • 可憐

    4

1.如文章侵犯了您的版權,請發郵件通知本站,該文章將在24小時内刪除;
2.本站標注原創的文章,轉發時煩請注明來源;
3.交流群: 2702237 13835667

相關課文
  • mac開發接入微信公衆號接口返回報錯 cURL error 56: SSLRead() return error -9806

  • pecl安裝程序時報錯Array and string offset access syntax with curly braces is no longer supported

  • PHP的換行符是什麽

  • 由於商家傳入的H5交易參數有誤,該筆交易暫時無法完成,請聯繫商家解決

我要說說
網上賓友點評
1 樓 IP 114.254.***.129 的嘉賓 说道 : 很久前
得救了

2 樓 IP 112.97.***.22 的嘉賓 说道 : 很久前
能具体讲讲到底改动了哪里了吗

3 樓 IP 222.168.***.253 的嘉賓 说道 : 很久前
不吹不黑, 这个版本的tp占据了国内php小半江山