php 5.3之前使用的垃圾回收機制是單純的“引用計數”,也就是每個內存對象都分配一個計數器,當內存對象被變量引用時,計數器 1;當變量引用撤掉后,計數器-1;當計數器=0時,表明內存對象沒有被使用,該內存對象則進行銷毀,垃圾回收完成。
“引用計數”存在問題,就是當兩個或多個對象互相引用形成環狀后,內存對象的計數器則不會消減為0;這時候,這一組內存對象已經沒用了,但是不能回收,從而導致內存泄露;
php5.3開始,使用了新的垃圾回收機制,在引用計數基礎上,實現了一種復雜的算法,來檢測內存對象中引用環的存在,以避免內存泄露。
該算法可以參考下面這篇文章,這是這篇小總結的主要參考文獻,淺談PHP5中垃圾回收算法(Garbage Collection)的演化.
看下面的例子,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- echo $b ."n";
- ?>
- //不用說 % php -f gc.php 輸出結果非常明了:
- hy0kl% php -f gc.php
- I am test.
好,下一個,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- $b = 'I will change?';
- //開源代碼Vevb.com
- echo $a ."n";
- echo $b ."n";
- ?>
- //執行結果依然很明顯:
- hy0kl% php -f gc.php
- I will change?
- I will change?
君請看,Example 3,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- unset($a);
- echo $a ."n";
- echo $b ."n";
- ?>
是不是得想一下下呢?
hy0kl% php -f gc.php
Notice: Undefined variable: a in /usr/local/www/apache22/data/test/gc.php on line 8
I am test.
有點犯迷糊了嗎?
君再看:Example 4,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- unset($b);
- echo $a ."n";
- echo $b ."n";
- ?>
其實如果 Example 3 理解了,這個與之異曲同工.
hy0kl% php -f gc.php
I am test.
Notice: Undefined variable: b in /usr/local/www/apache22/data/test/gc.php on line 9
君且看,Example 5,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- $a = null;
- echo '$a = '. $a ."n";
- echo
- '$b = '. $b ."n";
- ?>
猛的第一感覺是什么樣的?
hy0kl% php -f gc.php
$a =
$b =
沒錯,這就是輸出結果,對 PHP GC 已有深入理解的 phper 不會覺得有什么奇怪,說實話,當我第一次運行這段代碼時很意外,卻讓我對 PHP GC 有更深刻的理解了.那么下面與之同工的例子自然好理解了.
Example 6,代碼如下:
- <?php
- error_reporting(E_ALL);
- $a = 'I am test.';
- $b = & $a;
- $b = null;
- echo '$a = '. $a ."n";
- echo '$b = '. $b ."n";
- ?>
新聞熱點
疑難解答