PHP类型转换相关的一个Bug
浏览:1967次 出处信息
PHP为了避免数字索引和数字字符串索引(注1)的混乱, 引入了zend_symtable_*系列函数, 并应用于数组中.
这样一来, 数字字符串索引也就会被当作数字索引, 然而总是有一些情况, 是PHP的维护者没有想到的…
比如, 类型转换时刻:
- 鉴于很多朋友好心的提示, 使用json_deocde的第二个参数就可以直接得到数组.
- 我说明下, 如下的代码是我有意而为之, 并不是为了json_decode, 而是为了构造一个"有问题"的数组
在PHP5.2.*下(json version 1.2.1):
- $data = array(
- 123 => 'laruence',
- );
- $value = json_encode($data);
- $obj = json_decode($value);
- $arr = (array)$obj;
- var_dump($arr);
此时, 问题就出现了, 上面得到的输出是:
- array(1) {
- ["123"]=>
- string(8) "laruence"
现在,你郁闷吧, 因为数组键是字符串, 而通过正常渠道访问的时候, PHP都会自动把数字字符串转换成数字, 所以:
- print_r($arr[123]);
- //PHP Notice: Undefined offset: 123 in ***
- print_r($arr["123"]);
- //PHP Notice: Undefined offset: 123 in ***
- var_dump(array_key_exists("123", $arr));
- //bool(false)
我已经报了Bug, 不过PHP本身也不保证类型转换的一致性, 所以PHP维护者最后认为是不是Bug都无所谓了, 大家平时注意即可:http://bugs.php.net/bug.php?id=51915
注1
本文中所说的字符串数字和之前的文章PHP字符串比较中所说的numeric string有一点不同, 在zend_symtable_*系列函数中, 只会吧/^-?[^0][0-9]*$/这样的字符串认为是数字字符串. 相关核心逻辑如下:
- #define HANDLE_NUMERIC(key, length, func) {
- register char *tmp=key;
- if (*tmp=='-') {
- tmp++;
- }
- if ((*tmp>='0' && *tmp<='9')) do {
- char *end=key+length-1;
- long idx;
- if (*tmp++=='0' && length>2) {
- break;
- }
- while (tmp<end) {
- if (!(*tmp>='0' && *tmp<='9')) {
- break;
- }
- tmp++;
- }
- if (tmp==end && *tmp=='0') {
- if (*key=='-') {
- idx = strtol(key, NULL, 10);
- if (idx!=LONG_MIN) {
- return func;
- }
- } else {
- idx = strtol(key, NULL, 10);
- if (idx!=LONG_MAX) {
- return func;
- }
- }
- }
- } while (0);
- }
PS:我这里只有5.2.8, 5.2.11俩个版本, 各位读者如果有其他版本的PHP, 帮忙测试下是否在你的版本下也存在这个问题. 谢谢
另: 谢谢远豪提供这个问题, 原问题是和Memcached相关的.
建议继续学习:
- STRUTS2类型转换错误导致OGNL表达式注入漏洞分析 (阅读:9118)
- PHP数据类型隐性转换的陷阱 (阅读:2950)
- javascript运算/转换技巧 (阅读:2708)
- JavaScript ( (__ = !$ + $)[+$] + ({} + $)[_/_] +({} + $)[_/_] ) (阅读:2589)
- 编写安全代码:再论整数类型转换 (阅读:2619)
- 弱类型?C语言参数提升带来的一个陷阱 (阅读:2150)
- 当类型转换表达式遇上自定义转换操作 (阅读:1506)
QQ技术交流群:445447336,欢迎加入!
扫一扫订阅我的微信号:IT技术博客大学习
扫一扫订阅我的微信号:IT技术博客大学习
<< 前一篇:Ubuntu 下为 PHP 添加 Xdebug 插件
后一篇:使用PHP_UML生成代码的UML图 >>
文章信息
- 作者:雪候鸟 来源: 风雪之隅
- 标签: 类型转换
- 发布时间:2010-06-02 23:09:14
建议继续学习
近3天十大热文
- [68] Go Reflect 性能
- [68] 如何拿下简短的域名
- [67] Oracle MTS模式下 进程地址与会话信
- [62] IOS安全–浅谈关于IOS加固的几种方法
- [61] 图书馆的世界纪录
- [60] 【社会化设计】自我(self)部分――欢迎区
- [58] android 开发入门
- [56] 视觉调整-设计师 vs. 逻辑
- [49] 给自己的字体课(一)——英文字体基础
- [48] 读书笔记-壹百度:百度十年千倍的29条法则