溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

PHP垃圾回收機(jī)制如何對內(nèi)存泄露進(jìn)行處理

發(fā)布時(shí)間:2021-10-13 09:27:27 來源:億速云 閱讀:118 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下PHP垃圾回收機(jī)制如何對內(nèi)存泄露進(jìn)行處理,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

代碼如下:


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


在老的PHP版本中,這里就會(huì)出現(xiàn)內(nèi)存泄露,分析如下:

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

然后執(zhí)行第二行,$a將會(huì)從符號(hào)表中被刪除,同時(shí)指向的zval的refcount--,此時(shí)refcount=1,因?yàn)閞efcount!=0,故此zval不會(huì)被當(dāng)做垃圾回收,但是此時(shí)我們卻失去了$a[2]指向這個(gè)zval的入口,因此這個(gè)zval成了一塊內(nèi)存垃圾

同樣的道理可以發(fā)生在類內(nèi)部引用里,例如

復(fù)制代碼 代碼如下:


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


那么如何解決這種問題呢,新的GC機(jī)制采用了一個(gè)算法來解決這個(gè)問題

PHP有一個(gè)root buffer用來存儲(chǔ)zval的節(jié)點(diǎn)信息,當(dāng)root buffer滿了或者手動(dòng)調(diào)用gc函數(shù)時(shí),GC算法啟動(dòng)

對于一個(gè)數(shù)組或者類類型的zval而言,在垃圾回收機(jī)制啟動(dòng)時(shí),算法會(huì)對該zval的數(shù)組/類內(nèi)部的元素/成員的zval進(jìn)行一次遍歷并將refcount減1,如果說遍歷完成后該zval的refcount被減為0,則說明這個(gè)zval是一個(gè)內(nèi)存垃圾,他將被銷毀,見下面的例子

復(fù)制代碼 代碼如下:


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


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

當(dāng)unset($a)執(zhí)行的時(shí)候,$a就已經(jīng)從符號(hào)表中刪去,同時(shí)我們也失去了訪問z1的入口,此時(shí)z1 refcount=2,is_ref=1

當(dāng)GC啟動(dòng)時(shí),會(huì)對該z1的數(shù)組元素的zval的refcount進(jìn)行遍歷減1,遍歷到a[2]時(shí),z1 refcount--, a[3]時(shí) z1 refcount--,此時(shí)z1 refcount = 0,即可將z1標(biāo)記為內(nèi)存垃圾,算法后將其回收

總結(jié)來說可以這么表述:若一個(gè)數(shù)組類型的zval,對他的元素zval進(jìn)行一次遍歷,同時(shí)將遍歷到的zval的refcount--,如果最后refcount=0的zval,就是垃圾,需要被回收

以上是“PHP垃圾回收機(jī)制如何對內(nèi)存泄露進(jìn)行處理”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI