今天想利用利用jQuery的ajax和服务器语言php进行交互练习瀑布流,结果在交互过程中,不知哪一步使得php获取的字符串存在,可转为数组却是空,我天,这足足浪费我三个小时找这个bug,把每一个字符进行了一一比对,发现竟然语法都没错!!!最后的坑竟然在BOM头,好啦,把心情回归平静,把出现的问题,尝试的方法进行总结一下:
在本文中,$jsonStr代表php获取到的json字符串,$jsonArr代表php的json_decode()转化后的数组
1.首先php中的json_decode要求的字符串比较严格,以下几点需注意:
(1)使用UTF-8编码
(2)不能在最后元素有逗号
(3)不能使用单引号
(4)不能有\r,\t,如果有请替换
可以在点击打开链接上在线查询json格式正确与否
2.如果上述几点你都做到了,但是json_decode()的结果依然为空, 尝试
$jsonStr = preg_replace('/[\x00-\x1F\x80-\x9F]/u', '', trim($jsonStr));
以去掉其他破坏json字符串的字符。
3.可能因为在 json 字符串中反斜杠被转义,只需要用 htmlspecialchars_decode() 函数处理一下 :
$jsonStr = htmlspecialchars_decode($jsonStr);
然后再使用 json_decode() 函数解析得到$jsonArr
4.编码问题,尝试如下方法进行转码:
(1)在保存 json 数据时使用 urlencode() 函数:
$jsonStr= urlencode(json_encode($jsonStr));
解析时使用 urldecode() 函数:
$jsonStr = urldecode($jsonStr);
(2)
$jsonStr= stripslashes(html_entity_decode($jsonStr);
$jsonArr= json_decode($jsonStr,TRUE);
html_entity_decode() 函数的作用是把 HTML 实体转换为字符。
5.在转为json之前对中文进行转码:
function arrayRecursive(&$array, $function, $apply_to_keys_also = false)
{
static $recursive_counter = 0;
if (++$recursive_counter >1000 ) {
die('possible deep recursion attack');
}
foreach ($array as $key => $value) {
if (is_array($value)) {
arrayRecursive($array[$key], $function, $apply_to_keys_also);
} else {
$array[$key] = $function($value);
}
if ($apply_to_keys_also && is_string($key)) {
$new_key = $function($key);
if ($new_key != $key) {
$array[$new_key] = $array[$key];
unset($array[$key]);
}
}
}
$recursive_counter--;
}
/**************************************************************
*
* 将数组转换为JSON字符串(兼容中文)
* @param array $array 要转换的数组
* @return string 转换得到的json字符串
* @access public
*
*************************************************************/
function JSON($array)
{
arrayRecursive($array, 'urlencode', true);
$json = json_encode($array);
return urldecode($json);
}
echo JSON($jsonStr);
6.尝试在json_decode()的第二个参数中添加true
$jsonArr=json_decode($jsonStr,true)
7.也是我在交互中出现的问题,不小心在返回的json字符串中返回了BOM头的不可见字符,某些编辑器默认会加上BOM头,如下处理才能正确解析json数据:
$jsonArr= json_decode(trim($jsonStr,chr(239).chr(187).chr(191)),true);