PHP 遞歸效率分析
更新時(shí)間:2009年11月24日 21:35:34 作者:
PHP的遞歸效率一般認(rèn)為是低效的。大概一年前,我寫了一篇博文,對(duì)三種遍歷樹(shù)的方法進(jìn)行了比較,發(fā)現(xiàn)遞歸算法的效率最低。
而且是差了3倍的效率。所以,PHP中的遞歸一定要小心的對(duì)待。
最近寫了一個(gè)快速排序的算法,發(fā)現(xiàn)PHP中的遞歸效率不能一刀切,在各種不同的服務(wù)器中,可能會(huì)表現(xiàn)不一樣。
function qsort(&$arr)
{
_quick_sort($arr, 0, count($arr) - 1);
}
/**
* 采用遞歸算法的快速排序。
*
* @param array $arr 要排序的數(shù)組
* @param int $low 最低的排序子段
* @param int $high 最高的排序字段
*/
function _quick_sort(&$arr, $low, $high)
{
$low_data = $arr[$low];
$prev_low = $low;
$prev_high = $high;
while ($low < $high)
{
while ($arr[$high] >= $low_data && $low < $high) {
$high--;
}
if ($low < $high) {
$arr[$low] = $arr[$high];
$low++;
}
while ($arr[$low] <= $low_data && $low < $high) {
$low++;
}
if ($low < $high) {
$arr[$high] = $arr[$low];
$high--;
}
}
$arr[$low] = $low_data;
if ($prev_low < $low) {
_quick_sort($arr, $prev_low, $low);
}
if ($low + 1 < $prev_high) {
_quick_sort($arr, $low + 1, $prev_high);
}
}
function quick_sort(&$arr)
{
$stack = array();
array_push($stack, 0);
array_push($stack, count($arr) -1);
while (!empty($stack)) {
$high = array_pop($stack);
$low = array_pop($stack);
$low_data = $arr[$low];
$prev_low = $low;
$prev_high = $high;
while ($low < $high)
{
while ($arr[$high] >= $low_data && $low < $high) {
$high--;
}
if ($low < $high) {
$arr[$low] = $arr[$high];
$low++;
}
while ($arr[$low] <= $low_data && $low < $high) {
$low++;
}
if ($low < $high) {
$arr[$high] = $arr[$low];
$high--;
}
}
$arr[$low] = $low_data;
if ($prev_low < $low) {
array_push($stack, $prev_low);
array_push($stack, $low);
}
if ($low + 1 < $prev_high) {
array_push($stack, $low + 1);
array_push($stack, $prev_high);
}
}
}
下面是測(cè)試速度的代碼:
function qsort_test1()
{
$arr = range(1, 1000);
shuffle($arr);
$arr2 = $arr;
$t1 = microtime(true);
quick_sort($arr2);
$t2 = microtime(true) - $t1;
echo "非遞歸調(diào)用的花費(fèi):" . $t2 . "\n";
$arr1 = $arr;
$t1 = microtime(true);
qsort($arr1);
$t2 = microtime(true) - $t1;
echo "遞歸調(diào)用的花費(fèi):" . $t2 . "\n";
}
在我的IIS 服務(wù)器上(CGI)模式,我的測(cè)試結(jié)果是:
非遞歸調(diào)用的花費(fèi):0.036401009559631
遞歸調(diào)用的花費(fèi):0.053439617156982
在我的Apache 服務(wù)器上,我的測(cè)試結(jié)果是:
非遞歸調(diào)用的花費(fèi):0.022789001464844
遞歸調(diào)用的花費(fèi):0.014809131622314
結(jié)果完全相反,而PHP的版本是一樣的。
看來(lái)對(duì)遞歸的效率要具體問(wèn)題具體分析了。
最近寫了一個(gè)快速排序的算法,發(fā)現(xiàn)PHP中的遞歸效率不能一刀切,在各種不同的服務(wù)器中,可能會(huì)表現(xiàn)不一樣。
復(fù)制代碼 代碼如下:
function qsort(&$arr)
{
_quick_sort($arr, 0, count($arr) - 1);
}
/**
* 采用遞歸算法的快速排序。
*
* @param array $arr 要排序的數(shù)組
* @param int $low 最低的排序子段
* @param int $high 最高的排序字段
*/
function _quick_sort(&$arr, $low, $high)
{
$low_data = $arr[$low];
$prev_low = $low;
$prev_high = $high;
while ($low < $high)
{
while ($arr[$high] >= $low_data && $low < $high) {
$high--;
}
if ($low < $high) {
$arr[$low] = $arr[$high];
$low++;
}
while ($arr[$low] <= $low_data && $low < $high) {
$low++;
}
if ($low < $high) {
$arr[$high] = $arr[$low];
$high--;
}
}
$arr[$low] = $low_data;
if ($prev_low < $low) {
_quick_sort($arr, $prev_low, $low);
}
if ($low + 1 < $prev_high) {
_quick_sort($arr, $low + 1, $prev_high);
}
}
function quick_sort(&$arr)
{
$stack = array();
array_push($stack, 0);
array_push($stack, count($arr) -1);
while (!empty($stack)) {
$high = array_pop($stack);
$low = array_pop($stack);
$low_data = $arr[$low];
$prev_low = $low;
$prev_high = $high;
while ($low < $high)
{
while ($arr[$high] >= $low_data && $low < $high) {
$high--;
}
if ($low < $high) {
$arr[$low] = $arr[$high];
$low++;
}
while ($arr[$low] <= $low_data && $low < $high) {
$low++;
}
if ($low < $high) {
$arr[$high] = $arr[$low];
$high--;
}
}
$arr[$low] = $low_data;
if ($prev_low < $low) {
array_push($stack, $prev_low);
array_push($stack, $low);
}
if ($low + 1 < $prev_high) {
array_push($stack, $low + 1);
array_push($stack, $prev_high);
}
}
}
下面是測(cè)試速度的代碼:
復(fù)制代碼 代碼如下:
function qsort_test1()
{
$arr = range(1, 1000);
shuffle($arr);
$arr2 = $arr;
$t1 = microtime(true);
quick_sort($arr2);
$t2 = microtime(true) - $t1;
echo "非遞歸調(diào)用的花費(fèi):" . $t2 . "\n";
$arr1 = $arr;
$t1 = microtime(true);
qsort($arr1);
$t2 = microtime(true) - $t1;
echo "遞歸調(diào)用的花費(fèi):" . $t2 . "\n";
}
在我的IIS 服務(wù)器上(CGI)模式,我的測(cè)試結(jié)果是:
非遞歸調(diào)用的花費(fèi):0.036401009559631
遞歸調(diào)用的花費(fèi):0.053439617156982
在我的Apache 服務(wù)器上,我的測(cè)試結(jié)果是:
非遞歸調(diào)用的花費(fèi):0.022789001464844
遞歸調(diào)用的花費(fèi):0.014809131622314
結(jié)果完全相反,而PHP的版本是一樣的。
看來(lái)對(duì)遞歸的效率要具體問(wèn)題具體分析了。
相關(guān)文章
基于PHP創(chuàng)建Cookie數(shù)組的詳解
本篇文章是對(duì)在PHP中創(chuàng)建Cookie數(shù)組的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-07-07PHP字符轉(zhuǎn)義相關(guān)函數(shù)小結(jié)(php下的轉(zhuǎn)義字符串)
PHP字符轉(zhuǎn)義相關(guān)函數(shù)小結(jié),有時(shí)候?yàn)榱税踩鹨?jiàn),我們需要對(duì)用戶輸入的字符串進(jìn)行轉(zhuǎn)義2007-04-04PHP基于socket實(shí)現(xiàn)的簡(jiǎn)單客戶端和服務(wù)端通訊功能示例
這篇文章主要介紹了PHP基于socket實(shí)現(xiàn)的簡(jiǎn)單客戶端和服務(wù)端通訊功能,可實(shí)現(xiàn)服務(wù)端接收客戶端發(fā)送的字符串進(jìn)行翻轉(zhuǎn)操作后返回客戶端的功能,需要的朋友可以參考下2017-07-07PHP創(chuàng)建單例后臺(tái)進(jìn)程的方法示例
這篇文章主要介紹了PHP創(chuàng)建單例后臺(tái)進(jìn)程的方法,涉及php針對(duì)進(jìn)程的啟動(dòng)、創(chuàng)建、判斷、停止等相關(guān)操作技巧,需要的朋友可以參考下2017-05-05PHP進(jìn)階學(xué)習(xí)之Geo的地圖定位算法詳解
這篇文章主要介紹了PHP進(jìn)階學(xué)習(xí)之Geo的地圖定位算法,結(jié)合實(shí)例形式詳細(xì)分析了php Geo的地圖定位算法相關(guān)概念、原理、實(shí)現(xiàn)方法與操作注意事項(xiàng),需要的朋友可以參考下2019-06-06PHP MemCached 高級(jí)緩存應(yīng)用代碼
PHP MemCached 高級(jí)緩存應(yīng)用,使用MemCached的學(xué)習(xí)的朋友可以參考下。2010-08-08