之前同事在本地开发的时候,出现一个错误,如下图所示:
字面意思就是反序列化错误,由此bug引申出来序列化和反序列化得应用,以及php array当key为string类型的数字值时,会发生什么情形。
先来看序列化
$str = [
1 => 2,
'str' => '字符串'
];
var_dump( serialize($str) );
//结果:serialize.php:7:string 'a:2:{i:1;i:2;s:3:"str";s:9:"字符串";}' (length=40)
//其中第一个a 表示数据是个array 2表明长度为2
//{} 中表示数组内容
//i:1;i:2
// i表示为key为int类型 1表示key值为1
// i表示为value为int类型 2表示value值为2
//s:3:"str";s:9:"字符串"
// s表示key为string类型 3表示key长度为3 "str"表示value值为"str"
// s表示value为string类型 9表示value长度为9 "字符串"表示value值为"字符串"
第二点PHP中数组,会把在常量PHP_INT_MAX(int最大值)范围内的string类型的数值key自动转换成int类型。若数组的key超出常量PHP_INT_MAX的值,则此key仍然保持string类型,反之,被转换成int类型。
示例代码如下:
var_dump( PHP_INT_MAX );
$str = [
'1' => 111,
'9223372036854775808' => 'PHP_INT_MAX + 1'
];
var_dump($str);
输出结果如下图:
可以看到 key为'1'时在输出时变成了 1,而大于PHP_INT_MAX值得key依然是字符串。
回到标题反序列化错误的bug。
同事在本地开发的PHP_INT_MAX与测试环境不同,且在本地开发环境与测试环境存储在同一数据中。这个时候,就会出现报错的情况。即在本地开发中,可能PHP_INT_MAX较小,序列化会将数值转换成s,而测试环境PHP_INT_MAX较大,序列化会将数值转换成i。这个时候存储不统一,就会造成在一个环境中序列化,在另一个环境反序列化,报标题的错误。