您好,登錄后才能下訂單哦!
小編給大家分享一下PHP垃圾回收機制如何對內存泄露進行處理,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
代碼如下:
$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,就是垃圾,需要被回收
以上是“PHP垃圾回收機制如何對內存泄露進行處理”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。