1. 程式人生 > 程式設計 >PHP中垃圾回收相關函式的使用

PHP中垃圾回收相關函式的使用

之前我們已經學習過 php 中的引用計數以及垃圾回收機制的概念。這些內容非常偏理論,也是非常常見的面試內容。而今天介紹的則是具體的關於垃圾回收的一些功能函式。關於之前的兩篇介紹文章,大家可以到文章底部檢視。

再談迴圈引用以及強制清理迴圈引用

我們為什麼要強調 “迴圈引用” 呢?其實,在預設情況下,我們直接 unset() 掉一個沒有被其他變數引用的變數時,就會讓這個變數的引用計數變為0。這時,PHP 預設的垃圾回收機制就會直接清除掉這個變數。比如:

$a = new stdClass;
$b = new stdClass;
$c = new stdClass;
echo memory_get_usage(),PHP_EOL; // 706528

unset($a);
echo memory_get_usage(),PHP_EOL; // 706488

gc_collect_cycles();
echo memory_get_usage(),PHP_EOL; // 706488

從上面的程式碼中可以看出,我們 unset() 掉 $a 之後,記憶體直接就減少了。但是,如果是產生了迴圈引用的情況,那麼簡單的進行 unset() 就沒有效果了。

class D{
    public $d;
}
$d = new D;
$d->d = $d;
echo memory_get_usage(),PHP_EOL; // 706544

unset($d);
echo memory_get_usage(),PHP_EOL; // 706544

gc_collect_cycles();
echo memory_get_usage(),PHP_EOL; // 706488

在這段程式碼中,我們對 \$d 進行了一個簡單的迴圈引用賦值。使用 unset() 後,記憶體沒有發生變化,這時,只能使用 gc_collect_cycles() 函式來進行強制的迴圈引用清理,才能將 $d 裡面的無效迴圈引用清除掉。

沒錯,這一段的重點正是 gc_collect_cycles() 這個函式。它在正常情況下對普通的變數引用是不會產生什麼清理效果的,當然,對於普通的變數我們直接 unset() 掉就可以了。它最主要的作用就是針對迴圈引用的清理。之前我們學習過,迴圈引用計數會存在一個 根緩衝區 ,一般預設情況下它能容納 10000 個待清理的 可能根 。而 gc_collect_cycles() 的作用就是不用等這個 根緩衝區 滿就直接進行清理(個人理解)。關於這個垃圾回收演算法的內容請移步:PHP垃圾回收機制的一些淺薄理解

其實,大部分情況下我們是不太需要關注 PHP 的垃圾回收問題的,也就是說,我們不是很需要手動地去呼叫這個 gc_collect_cycles() 函式。PHP-FPM 在每次呼叫完成後會直接整體的釋放,簡單的一次 CLI

指令碼執行完也會全部釋放。沒錯,正常情況下,PHP 一次執行完成之後就會銷燬所有的內容,記憶體垃圾自然也就不存在了。但是,在執行長時間的守護指令碼時,或者使用常駐程序的框架(Swoole)時,還是需要注意有沒有迴圈引用的問題。因為這種程式一直執行,如果存在大量迴圈引用物件時,就有可能導致記憶體洩露。

開啟、關閉及檢視迴圈引用垃圾回收狀態

gc_disable();
echo gc_enabled(),PHP_EOL; //
gc_enable();
echo gc_enabled(),PHP_EOL; // 1
程式設計客棧

很簡單的三個函式,gc_disable() 是 “停用迴圈引用收集器”,gc_enable() 是“開啟循http://www.cppcns.com環引用收集器”,而 gc_enabled() 就是檢視當前的迴圈引用收集器是否開啟。

強制回收Zend引擎記憶體管理器使用的記憶體

gc_mem_caches()

官網及網路上並沒有什麼詳細的介紹,不過從定義來看,它主要的作用就是回收 PHP 底層的 Zend 引擎記憶體管理器所使用過的記憶體。這個大家瞭解下就好,平常也從來沒用過。

獲取垃圾收http://www.cppcns.com集器的資訊

$e = new stdClass;
for($i = 100;$i>0;$i--){
    $e->list[] = $e;
}

unset($e);
gc_collect_cycles();

var_dNHZGRump(gc_status());
// array(4) {
//     ["runs"]=>int(1)
//     ["collected"]=>int(2)
//     ["threshold"]=>int(10001)
//     ["roots"]=>int(0)
// }

我們還是做了一個迴圈引用的物件,然後使用 gc_status() 來檢視當前垃圾回收器中關於迴圈引用的狀態。從返回的內容可以看出, runs 運行了 1 個,collected 收集了 2 個, thresNHZGRhold 閾值是 10001,roots 可能根沒有了(已經被回收了)。

這個函式可以在測試環境中對程式碼的執行情況進行檢查,檢視我們程式碼中有沒有不正常的迴圈引用情況,當然,上面的解釋也只是個人的推測,因為關於這方面的資料確實非常少。所以也希望深入研究過這方面內容的大神能夠留言指點迷津!!

測試程式碼:

github.com/zhangyue050…

相關文章

//www.jb51.net/article/210943.htm

//www.jb51.net/article/210957.htm

以上就是PHP中垃圾回收相關函式的使用的詳細內容,更多關於PHP中垃圾回收相關函式的資料請關注我們其它相關文章!