深入PHP-直观观察array的扩展

皆知PHP的数组是由HashTable和双链表实现的,为了方便大家查看数组的数据结构,开发一个遍历PHP的数组生成Dot描述的PHP插件,生成dot描述以后可以通过一些渲染工具生成图像,本例用的是 Graphviz。
扩展的实现很简单,PHP数组源码是由下面的两种结构体实现的,扩展就是将这两种结构体和各个结构体的关系遍历一遍,生成对应的Dot描述即可。
typedef struct bucket {
ulong h;
uint nKeyLength;
void * pData;
void * pDataPtr;
struct bucket * pListNext;
struct bucket * pListLast;
struct bucket *pNext ;
struct bucket * pLast;
const char *arKey ;
} Bucket;

typedef struct _hashtable {
uint nTableSize;
uint nTableMask;
uint nNumOfElements;
ulong nNextFreeElement;
Bucket *pInternalPointer;
Bucket *pListHead;
Bucket *pListTail;
Bucket **arBuckets;
dtor_func_t pDestructor;
zend_bool persistent;
unsigned char nApplyCount ;
zend_bool bApplyProtection;
} HashTable;



[size=large][b]扩展里边函数说明[/b][/size]
---------------------------------------------------------------------------------------------
string dotarray( array $input [, int $flag] )
生成数组数据结构的dot描述
[b]参数[/b]
input 需要操作的数组
flag 查看那些数据结构,是扩展提供的3个常量或操作,分别是
DOTARAAY_HASH_TABLE 表示显示HashTable结构
DOTARRAY_DOUBLE_LIST 显示数组的双链表结构
DOTARRAY_CURRENT_POSITION 显示数组现在的内部指针的位置
[b]返回值[/b]
成功返回dot描述字符串,错误(非数组等情况)返回false
---------------------------------------------------------------------------------------------
[b][size=large]显示双链表结构例子[/size][/b]

<?php

$items = array(1,2,8=>'lalala',16=>'hahaha','name'=>'shiki',30=>'wooooo...');
next($items);/*将内部指针移到下一位*/
$result = dotarray($items,DOTARRAY_DOUBLE_LIST|DOTARRAY_CURRENT_POSITION);
echo $result;

得到的dot描述:

digraph html {label = "Structure of array";
node[shape = record];
===========部分内容省略============
edge [color=black];
edge [color=green];
sk_array:f5:s -> sk_item_1:f0;
edge [color=black];
}

通过graphviz渲染得到下面的图片

[img]http://dl2.iteye.com/upload/attachment/0107/2260/10f186b3-e037-3772-93d3-ea06609e6c89.png[/img]


需要解释一下,红色实线为结构中双链表的next指针,红色点线为上一个元素指针。同时提供了DOTARRAY_CURRENT_POSITION标志,出现绿线指向数组内部指针指向key为1的元素(第一个元素key为0)

[size=large][b]显示HashTable结构例子[/b][/size]

<?php

$items = array(1,2,8=>'lalala',16=>'hahaha','name'=>'shiki',30=>'wooooo...');
next($items);
next($items);
$result = dotarray($items,DOTARAAY_HASH_TABLE|DOTARRAY_CURRENT_POSITION);
echo $result;

生成的结构图片如下:

[img]http://dl2.iteye.com/upload/attachment/0107/2262/fb9a1b97-7cfc-3db6-b474-ba9b56fe5edb.png[/img]

[b][size=large]hash冲突的例子[/size][/b]
具体可以参考 PHP hash碰撞拒绝服务漏洞原理[url]http://www.laruence.com/2011/12/29/2412.html[/url],简单介绍即是:Hash表因为”冲突”(碰撞)而退化成链表,如果数据量足够大, 使得php在计算, 查找, 插入的时候, 造成大量的CPU占用, 从而实现拒绝服务攻击。下面是简单的碰撞生成代码:

<
?php

/*会生出pow(2,$n)个数字*/
$n= 3;
$capacity = pow(2,$n);
$array =array();
for($i=0;$i<$capacity;$i++){
$key = $i<<$n;
$array[$key] =1;
}


$str = dotarray($array,DOTARRAY_DOUBLE_LIST|DOTARAAY_HASH_TABLE);
echo $str,PHP_EOL;

这次将hashtable和双线表同时展示出来,传递的flag为DOTARRAY_DOUBLE_LIST|DOTARAAY_HASH_TABLE,同时展示双链表和HashTable

[img]http://dl2.iteye.com/upload/attachment/0107/2430/cdc58e24-d88a-354f-a772-aa7fbb1676da.png[/img]

关于 dot语言 传送门:http://zh.wikipedia.org/wiki/DOT%E8%AF%AD%E8%A8%80
关于 dot渲染工具Graphviz 传送门:http://www.graphviz.org/
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值