一次有趣的 php json_decode error 分析

以前同事问了我一个有趣的问题,说为何这个 json 字符串从缓存里读出来能够解析,可是做为一个 php 变量却解析错误呢?php

先看一个我已经简化了的例子json

$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;

输出.net

NULL

Control character error, possibly incorrectly encoded

你就会发现,无论怎么搞,json_decode 老是失败的,而后我慢慢的把 json 字符串抽茧剥丝,最后只剩下“\\”,发现问题出在这里。code

乍一看,这个字符串并无什么特别之处,可是在我把它 echo 以后,终于搞明白到底为何一直 json_decode 失败了字符串

$conf = <<<EOD
"\\"
EOD;
echo $conf,PHP_EOL;

输出get

\

“\\” 这货被转义了,而后这就再也不是一个正常的 json 字符串了,因此 json_decode 就会失败,那么才能让它正确的被 decode 呢?看下面这段代码string

$conf = <<<'EOD'
"\\"
EOD; // nowdoc
var_dump(json_decode($conf, true));
echo json_last_error_msg(), PHP_EOL;

输出ast

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架构师

相关文章
相关标签/搜索