佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > 淺析PHP7 的垃圾回收機(jī)制

淺析PHP7 的垃圾回收機(jī)制

熱門標(biāo)簽:Linux服務(wù)器 電子圍欄 科大訊飛語音識別系統(tǒng) 服務(wù)器配置 Mysql連接數(shù)設(shè)置 阿里云 團(tuán)購網(wǎng)站 銀行業(yè)務(wù)

垃圾回收機(jī)制

垃圾回收機(jī)制是一種動態(tài)存儲分配方案。它會自動釋放程序不再需要的已分配的內(nèi)存塊。 自動回收內(nèi)存的過程叫垃圾收集。垃圾回收機(jī)制可以讓程序員不必過分關(guān)心程序內(nèi)存分配,從而將更多的精力投入到業(yè)務(wù)邏輯。 在現(xiàn)在的流行各種語言當(dāng)中,垃圾回收機(jī)制是新一代語言所共有的特征。

垃圾的產(chǎn)生

PHP7 中復(fù)雜類型,像字符串、數(shù)組、對象等的數(shù)據(jù)結(jié)構(gòu)中,頭部都有一個 gc, 這個 gc 的作用就是用來對垃圾回收的支持。當(dāng)變量賦值、傳遞時,會增加 value 的引用數(shù), unset、return 等釋放變量時再減掉引用數(shù),減掉后如果發(fā)現(xiàn) refcount 變?yōu)?0 則直接釋放 value,這是變量的基本回收過程。

不過有一種問題是這個機(jī)制無法解決的,就是循環(huán)引用的問題。

什么是循環(huán)引用呢? 簡單說就是變量的內(nèi)部里存的 value 又引用了變量自身。 這種比較經(jīng)常發(fā)生在數(shù)組和對象類型的變量上。

這里先講一下引用,即 zend_reference 這個類型,這個是 PHP7 新增的變量類型,當(dāng)對變量使用 “” 操作時,會創(chuàng)建新的中間結(jié)構(gòu)體 zend_reference,這個結(jié)構(gòu)體會真正的指向?qū)?yīng)的 value 結(jié)構(gòu)。

舉個例子:

// 當(dāng)進(jìn)行如下賦值操作時
$a = 'hello'; // $a -> zend_string
$b = $a; // $b,$a -> zend_string
$c = $b; // $c,$b -> zval(type = IS_REFERENCE, refcount = 2) -> zend_string


最終會變成如下這樣:

 

即 $b 和 $c 的 zval 是通過中間結(jié)構(gòu)體 zend_reference 再指向最終的 zend_string。

回到循環(huán)引用的問題,舉個數(shù)組循環(huán)引用例子:

$arr = [1];
$a[] = $a;
unset($a);

使用 操作之后,變量 a 就變成了引用類型且引用計數(shù) refcount 為 2,而又賦值給自己里面的元素,即變量 a 變成了自己引用自己。

具體如下如所示:

 

當(dāng) unset 之后就變成下圖這樣:

 

即 $a 所在的 zval 類型已經(jīng)變成了 IS_UNDEF 了,zend_reference 結(jié)構(gòu)體的引用計數(shù)減 1,但是仍然大于 0,這時候,這部分結(jié)構(gòu)體就變成了垃圾,對此不處理的話,就可能會造成內(nèi)存泄露。這里就需要垃圾收集器將這部分收集到緩沖區(qū),之后進(jìn)行回收處理。

回收過程

如果當(dāng)變量的 refcount 減小后大于 0,PHP 并不會立即對這個變量進(jìn)行垃圾鑒定和回收,而是放入一個緩沖區(qū)中,等這個緩沖區(qū)滿了以后(10000 個值)再統(tǒng)一進(jìn)行處理,加入緩沖區(qū)的是變量 zend_value 里的 gc,目前垃圾只會出現(xiàn)在數(shù)組和對象兩種類型中,數(shù)組的情況上面已經(jīng)介紹了,對象的情況則是成員屬性引用對象本身導(dǎo)致的,其它類型不會出現(xiàn)這種變量中的成員引用變量自身的情況,所以垃圾回收只會處理這兩種類型的變量。

gc 的結(jié)構(gòu) zend_refcounted_h 具體如下:

typedef struct _zend_refcounted_h {
  uint32_t     refcount; // 記錄 zend_value 的引用數(shù)
  union {
    struct {
      zend_uchar  type, // zend_value的類型, 與zval.u1.type一致
      zend_uchar  flags, 
      uint16_t   gc_info // GC信息,記錄在 gc 池中的位置和顏色,垃圾回收的過程會用到
    } v;
    uint32_t type_info;
  } u;
} zend_refcounted_h;

一個變量只能加入一次緩沖區(qū),為了防止重復(fù)加入,變量加入后會把 zend_refcounted_h.gc_info 置為 GC_PURPLE,即標(biāo)為紫色,后續(xù)不會重復(fù)插入。

垃圾緩沖區(qū)是一個雙向鏈表,等到緩存區(qū)滿了以后則啟動垃圾檢查過程:遍歷緩沖區(qū),對當(dāng)前變量的所有成員進(jìn)行遍歷,然后把成員的 refcount 減 1 (如果成員還包含子成員則也進(jìn)行遞歸遍歷,即深度優(yōu)先遍歷),最后再檢查當(dāng)前變量的引用,如果減為了 0 則為垃圾。這個算法的原理核心是:垃圾是由于成員引用自身導(dǎo)致的,那么就對所有的成員減一遍引用,如果發(fā)現(xiàn)最后變量本身的 refcount 變?yōu)榱?0 則就表明其引用全部來自自身成員,即其他任何地方都不再使用它,那么它就是垃圾,需要被回收掉。反之說明不是垃圾,需要將其從緩沖區(qū)移出去。具體的過程如下:

(1) 從緩沖區(qū)鏈表的 roots 開始遍歷,把當(dāng)前 value 標(biāo)為灰色 (zend_refcounted_h.gc_info 置為 GC_GREY),然后對當(dāng)前 value 的成員進(jìn)行深度優(yōu)先遍歷,把成員 value 的 refcount 減 1,并且也標(biāo)為灰色;

(2) 重復(fù)遍歷緩沖區(qū)鏈表,檢查當(dāng)前 value 引用是否為 0,為 0 則表示確實(shí)是垃圾,把它標(biāo)為白色(GC_WHITE),如果不為 0 則排除了引用全部來自自身成員的可能,表示還有外部的引用,并不是垃圾,這時候因?yàn)椴襟E(1)對成員進(jìn)行了 refcount 減 1 操作,需要再還原回去,對所有成員進(jìn)行深度遍歷,把成員 refcount 加 1,同時標(biāo)為黑色;

(3) 再次遍歷緩沖區(qū)鏈表,將非 GC_WHITE 的節(jié)點(diǎn)從 roots 鏈表中移出(移到待釋放的列表),最終 roots 鏈表中全部為真正的垃圾,最后將這些垃圾清除。

總結(jié)

以上所述是小編給大家介紹的PHP7 的垃圾回收機(jī)制,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
如果你覺得本文對你有幫助,歡迎轉(zhuǎn)載,煩請注明出處,謝謝!

您可能感興趣的文章:
  • PHP垃圾回收機(jī)制的一些理解
  • PHP的垃圾回收機(jī)制代碼實(shí)例講解
  • PHP session垃圾回收機(jī)制實(shí)例分析
  • PHP進(jìn)階學(xué)習(xí)之垃圾回收機(jī)制詳解
  • PHP析構(gòu)函數(shù)destruct與垃圾回收機(jī)制的講解
  • 掌握PHP垃圾回收機(jī)制詳解
  • php中session垃圾回收機(jī)制
  • 總結(jié)PHP內(nèi)存釋放以及垃圾回收
  • 解讀PHP中的垃圾回收機(jī)制
  • 析構(gòu)函數(shù)與php的垃圾回收機(jī)制詳解
  • PHP中垃圾回收相關(guān)函數(shù)的使用

標(biāo)簽:萍鄉(xiāng) 衢州 衡水 廣元 棗莊 江蘇 蚌埠 大理

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《淺析PHP7 的垃圾回收機(jī)制》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    龙门县| 拜城县| 万全县| 铁岭县| 崇文区| 嘉黎县| 咸丰县| 尼木县| 湘阴县| 利川市| 五峰| 夏津县| 达拉特旗| 西藏| 依兰县| 改则县| 达尔| 灌云县| 台州市| 永春县| 海盐县| 三明市| 辽源市| 阿拉善盟| 余姚市| 新营市| 阳泉市| 宾阳县| 民县| 长白| 丘北县| 三门峡市| 利川市| 黑山县| 景谷| 平度市| 阆中市| 水城县| 大港区| 永丰县| 莱阳市|