看PHP手冊的時候發現了下面這樣一段代碼:
復制代碼 代碼如下:
<?php
function Test()
{
static $count = 0;
$count++;
echo $count;
if ($count < 10) {
Test();
}
$count--;
}
?>
執行結果如下:
這是一個遞歸的函數,聲明的靜態變量count記錄次數,輸出1~10。
我在看的時候有個疑惑,遞歸調用的時候 static $count = 0; 語句會重復執行,這為什麼不會導致count變量被重復賦值呢?帶著這個疑問和同事研究了一下,測試用代碼如下:
復制代碼 代碼如下:
<?php
echo 'start<br />';
static $a = 10;
echo "$a <br />";
unset($GLOBALS['a']);
echo "$a <br />";
static $a = 20;
echo "$a <br />";
$GLOBALS['a'] = 10;
echo "$a <br />";
static $a = 30;
echo "$a <br />";
unset($GLOBALS['a']);
echo "$a <br />";
static $a;
echo "$a <br />";
static $a = 40;
echo "$a <br />";
$a = 100;
echo "$a <br />";
static $a = 50;
echo "$a <br />";
static $a = 4;
echo "$a <br />";
echo 'end <br />';
exit;
?>
執行結果如下:
start
- 4
- Notice: Undefined variable: a
- 4
- 10
- 10
- Notice: Undefined variable: a
- 10
- 10
- 100
- 100
- 100
- end
(結果中關於文件位置的部分已刪去。也可以去掉echo語句使用zend的debug功能查看,這樣結果更清晰)
代碼第5行第一次輸出$a的值為4,由此推測PHP在頁面初始化的時候分配靜態變量的內存,此時使用了同一個變量的最後一次聲明的值(這個可以把4改為其他數測試)。代碼第7行調用unset函數銷毀變量$a,再次輸出$a的值時看到未定義變量的提示,說明變量已經被銷毀。
第10行再次輸出時,輸出結果仍是4而不是20,有兩種可能,一個是php再次初始化了$a的值,另一種是php使用了$a被銷毀前的值,這個問題在第20行輸出的時候解決。第16行$a銷毀的時候值為10,第19行聲明後輸出仍為10。
第11行將$a的值修改為10,在14行再次聲明$a,17行輸出認為10。推測為重復聲明時php還是使用靜態變量內存中的值,而不再次賦值。
至此,手冊中發現的問題,大致上已經解決了,即遞歸調用中的聲明沒有改變$count的值,所以遞歸在$count=10時成功停止。
可能有理解不正確的地方,歡迎拍磚。