|
马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。
您需要 登录 才可以下载或查看,没有帐号?立即注册
x
告诉你了一个方式,但是缺少努力这一环节,那也是白搭。 PHP数组的Hash抵触实例,你晓得不晓得, 拔出65536个经由机关的键值的元素到PHP数组, 会需求耗时30秒以上? 而普通的这个进程仅仅需求0.1秒..
请看以下的例子:
$size = pow(2, 16);
$startTime = microtime(true);
$array = array();
for ($key = 0, $maxKey = ($size - 1) * $size; $key <= $maxKey; $key += $size) { $array[$key] = 0;
}$endTime = microtime(true);
echo '拔出 ', $size, ' 个歹意的元素需求 ', $endTime - $startTime, ' 秒', "\n"; $startTime = microtime(true);
$array = array();
for ($key = 0, $maxKey = $size - 1;
$key <= $maxKey; ++$key) { $array[$key] = 0;}$endTime = microtime(true);
echo '拔出 ', $size, ' 个通俗元素需求 ', $endTime - $startTime, ' 秒', "\n";
下面的例子, 在我的机械上的履行了局以下:
拔出 65536 个歹意的元素需求 43.1438360214 秒拔出 65536 个通俗元素需求 0.0210378170013 秒
这个不同是否是很夸大?!
我在上一篇文章中引见过, 经由特别机关的键值, 使得PHP每次拔出城市形成Hash抵触, 从而使得PHP中array的底层Hash表退步成链表:
Hash collision
如许在每次拔出的时分PHP都需求遍历一遍这个链表, 人人可以想象, 第一次拔出, 需求遍历0个元素, 第二次是1个, 第三次是3个, 第65536个是65535个, 那末总共就需求65534*65535/2=2147385345次遍历….
那末, 这个键值是怎样机关的呢?
在PHP中,假如键值是数字, 那末Hash的时分就是数字自己, 普通的时分都是, index & tableMask. 而tableMask是用来包管数字索引不会超越数组可包容的元素个数值, 也就是数组个数-1.
PHP的Hashtable的巨细都是2的指数, 好比假如你存入10个元素的数组, 那末数组实践巨细是16, 假如存入20个, 则实践巨细为32, 而63个话, 实践巨细为64. 当你的存入的元素个数大于了数组今朝的最多元素个数的时分, PHP会对这个数组停止扩容, 而且重新Hash.
如今, 咱们假定要存入64个元素(两头能够会经由扩容, 然而咱们只需求晓得, 最初的数组巨细是64, 而且对应的tableMask为63:0111111), 那末假如第一次咱们存入的元素的键值为0, 则hash后的值为0, 第二次咱们存入64, hash(1000000 & 0111111)的值也为0, 第三次咱们用128, 第四次用192… 就能够使得底层的PHP数组把一切的元素都Hash到0号bucket上, 从而使得Hash表退步成链表了.
固然, 假如键值是字符串的话, 就略微对照费事一些了, 然而PHP的Hash算法是开源的, 已知的, 所以有心人也能够做到…
本文地址: http://www.laruence.com/2011/12/30/2435.html
不断巩固,摸透大部分PHP常用函数,并可理解OOP,MYSQL优化,以及模板 |
|