很多程序員以爲現在時興的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; } }
至於其他問題, 你還有改不了的嗎? 請留言。
2 樓 IP 112.97.***.22 的嘉賓 说道 : 很久前
3 樓 IP 222.168.***.253 的嘉賓 说道 : 很久前