json_decode 整型溢出解决方法

这个有点像mysql int类型超过了就溢出,而我们只要用bigint就可以了,那么因为php中int数据范围的问题,所以就也有可能出现这类问题。

编码过程中遇到个错误,就是在处理json时,数值较大的int值在解码后数据被损坏,比如:

  1. $array = array(
  2. "id1" => 2147483647,
  3. "id2" => 2147483648
  4. );
  5. $json = json_encode($array);
  6. $out = json_decode($json, true);
  7. var_dump($out);
  8. 理论上应该看到:
  9. array(2) {
  10. ["id1"]=>int(2147483647)
  11. ["id2"]=>int(2147483648)
  12. }
  13. 但实际在我的电脑上却得到:
  14. 代码如下 复制代码
  15. array(2) {
  16. ["id1"]=>int(2147483647)
  17. ["id2"]=>int(-2147483646)
  18. }

这是由PHP整数值范围决定的,而这个范围依赖于操作系统。在32位操作系统中,PHP的整数最大值是2147483647,你可以通过输出PHP_INT_MAX看到。

一般情况下,你赋值一个很大的数,PHP会自动判定这个数值的范围并自动转换类型,如:

  1. $large_number = 2147483647;
  2. var_dump($large_number); // int(2147483647)
  3. $large_number = 2147483648;
  4. var_dump($large_number); // float(2147483648)
  5. $million = 1000000;
  6. $large_number = 50000 * $million;
  7. var_dump($large_number); // float(50000000000)

但是在json_decode方法中没有进行这种检测,这是PHP(旧版本)的bug,在5.3以后的版本,就不存在这个问题了。

如果你不想更新你的PHP,那还有个办法,就是将数字转为字符串。还是以上面的代码为例:

  1. $array = array(
  2. "id1" => 2147483647,
  3. "id2" => 2147483648
  4. );
  5. $json = json_encode($array);
  6. $json = preg_replace('/("idd":)(d{9,})/i', '${1}"${2}"', $json);
  7. $out = json_decode($json, true);
  8. var_dump($out);

当然,这个怎么替换是按需而定的,而且需要比较细致的测试。