深入解析PHP記憶體回收機制對記憶體泄露的處理_php技巧

來源:互聯網
上載者:User

上次說到了refcount和is_ref,這裡來說說記憶體泄露的情況

複製代碼 代碼如下:

$a = array(1, 2, &$a);
unset($a);

在老的PHP版本中,這裡就會出現記憶體泄露,分析如下:

執行第一行,可以知道$a和$a[2]指向的zval refcount=2,is_ref=1

然後執行第二行,$a將會從符號表中被刪除,同時指向的zval的refcount--,此時refcount=1,因為refcount!=0,故此zval不會被當做記憶體回收,但是此時我們卻失去了$a[2]指向這個zval的入口,因此這個zval成了一塊記憶體垃圾

同樣的道理可以發生在類內部引用裡,例如

複製代碼 代碼如下:

$a = new Man();
$a->self = &$a;
unset($a);

那麼如何解決這種問題呢,新的GC機制採用了一個演算法來解決這個問題

PHP有一個root buffer用來儲存zval的節點資訊,當root buffer滿了或者手動調用gc函數時,GC演算法啟動

對於一個數組或者類類型的zval而言,在記憶體回收機制啟動時,演算法會對該zval的數組/類內部的元素/成員的zval進行一次遍曆並將refcount減1,如果說遍曆完成後該zval的refcount被減為0,則說明這個zval是一個記憶體垃圾,他將被銷毀,見下面的例子

複製代碼 代碼如下:

$a = array(1, 2, &$a, &$a);
unset($a);

容易知道$a指向的zval,假設為z1的refcount=3,is_ref=1

當unset($a)執行的時候,$a就已經從符號表中刪去,同時我們也失去了訪問z1的入口,此時z1 refcount=2,is_ref=1

當GC啟動時,會對該z1的數組元素的zval的refcount進行遍曆減1,遍曆到a[2]時,z1 refcount--, a[3]時 z1 refcount--,此時z1 refcount = 0,即可將z1標記為記憶體垃圾,演算法後將其回收

總結來說可以這麼表述:若一個數群組類型的zval,對他的元素zval進行一次遍曆,同時將遍曆到的zval的refcount--,如果最後refcount=0的zval,就是垃圾,需要被回收

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.