之前同事问了我一个有趣的问题,说为什么这个 json 字符串从缓存里读出来可以解析,但是作为一个 php 变量却解析错误呢?
先看一个我已经简化了的例子
$str = "\\";
$j = json_encode($str);
echo $j, PHP_EOL;
echo json_decode($j), PHP_EOL;
输出
"\\"
\
如果你直接把输出的 json 字符串赋值给一个 php 字符串变量的话,好玩的事情就发生了
$conf = <<<EOD
"\\"
EOD; // heredoc
// $conf = '"\\"'; // 单引号
// $conf = "\"\\\""; // 双引号
var_dump(json_decode($conf, true));
echo json_last_error_msg(), PHP_EOL;
输出
NULL
Control character error, possibly incorrectly encoded
你就会发现,不管怎么搞,json_decode 总是失败的,然后我慢慢的把 json 字符串抽茧剥丝,最后只剩下“\\”,发现问题出在这里。
乍一看,这个字符串并没有什么特别之处,但是在我把它 echo 之后,终于搞明白到底为什么一直 json_decode 失败了
$conf = <<<EOD
"\\"
EOD;
echo $conf,PHP_EOL;
输出
\
“\\” 这货被转义了,然后这就不再是一个正常的 json 字符串了,所以 json_decode 就会失败,那么才能让它正确的被 decode 呢?看下面这段代码
$conf = <<<'EOD'
"\\"
EOD; // nowdoc
var_dump(json_decode($conf, true));
echo json_last_error_msg(), PHP_EOL;
输出
string(1) "\"
No error
我们使用 nowdoc (http://php.net/manual/zh/language.types.string.php#language.types.string.syntax.nowdoc),这样就不会被转义了。
就象 heredoc 结构类似于双引号字符串,Nowdoc 结构是类似于单引号字符串的。Nowdoc 结构很象 heredoc 结构,但是 nowdoc 中不进行解析操作。这种结构很适合用于嵌入 PHP 代码或其它大段文本而无需对其中的特殊字符进行转义。
更多架构、PHP、GO相关踩坑实践技巧请关注我的公众号:PHP架构师