主頁 > 知識(shí)庫 > PHP的垃圾回收機(jī)制代碼實(shí)例講解

PHP的垃圾回收機(jī)制代碼實(shí)例講解

熱門標(biāo)簽:常州地圖標(biāo)注服務(wù)商 地圖標(biāo)注平臺(tái)怎么給錢注冊 福州人工外呼系統(tǒng)哪家強(qiáng) 釘釘打卡地圖標(biāo)注 衡水外呼系統(tǒng)平臺(tái) 百度商鋪地圖標(biāo)注 安裝電銷外呼系統(tǒng) 新河科技智能外呼系統(tǒng)怎么樣 注冊400電話申請

PHP可以自動(dòng)進(jìn)行內(nèi)存管理,清除不需要的對象,主要使用了引用計(jì)數(shù)

zval結(jié)構(gòu)體中定義了ref_countis_ref , ref_count是引用計(jì)數(shù) ,標(biāo)識(shí)此zval被多少個(gè)變量引用 , 為0時(shí)會(huì)被銷毀
is_ref標(biāo)識(shí)是否使用的 取地址符強(qiáng)制引用

為了解決循環(huán)引用內(nèi)存泄露問題 , 使用同步周期回收算法
比如當(dāng)數(shù)組或?qū)ο笱h(huán)的引用自身 , unset掉數(shù)組的時(shí)候 , 當(dāng)refcount-1后還大于0的 , 就會(huì)被當(dāng)成疑似垃圾 , 會(huì)進(jìn)行遍歷 ,并且模擬的刪除一次refcount-1如果是0就刪除 ,如果不是0就恢復(fù)

頑固垃圾的產(chǎn)生過程

?php
  $a = "new string";
?>

代碼中,$a變量內(nèi)部存儲(chǔ)信息為

a: (refcount_gc=1, is_ref_gc=0)='new string'

當(dāng)把 a 賦 值 給 另 外 一 個(gè) 變 量 的 時(shí) 候 , a賦值給另外一個(gè)變量的時(shí)候, a賦值給另外一個(gè)變量的時(shí)候,a對應(yīng)的zval的refcount_gc會(huì)加1

?php
  $a = "new string";
  $b = $a;
?>

此時(shí) a 和 a和 a和b變量對應(yīng)的內(nèi)部存儲(chǔ)信息為, a 和 a和 a和b同時(shí)指向一個(gè)字符串"new string" ,它的refcount變成2

a,b: (refcount_gc=2, is_ref=0)='new string'

當(dāng)用unset刪除$b變量時(shí),“new string” 的refcount_gc會(huì)減1變成1。

?php
  $a = "new string"; //a: (refcount_gc=1, is_ref_gc=0)='new string'
  $b = $a;      //a,b: (refcount_gc=2, is_ref=0)='new string'
  unset($b);     //a: (refcount_gc=1, is_ref=0)='new string'
?>

對于普通的變量來說,這一切很正常,但是在復(fù)合類型變量(數(shù)組和對象)中,會(huì)發(fā)生比較有意思的事情:

?php
  $a = array('meaning' => 'life', 'number' => 42);
?>

$a內(nèi)部存儲(chǔ)信息為:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=1, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42
)

數(shù)組變量本身($a)在引擎內(nèi)部實(shí)際上是一個(gè)哈希表,這張表中有兩個(gè)zval項(xiàng) meaning和number,所以實(shí)際上那一行代碼中一共生成了3個(gè)zval,這3個(gè)zval都遵循變量的引用和計(jì)數(shù)原則,用圖來表示:

下面在$a中添加一個(gè)元素,并將現(xiàn)有的一個(gè)元素的值賦給新的元素:

?php
  $a = array('meaning' => 'life', 'number' => 42);
  $a['name'] = $a['meaning'];
?>

那么$a的內(nèi)部存儲(chǔ)為 , “l(fā)ife” 的ref_count變成2 , 42的ref_count是1:

a: (refcount=1, is_ref=0)=array (
'meaning' => (refcount=2, is_ref=0)='life',
'number' => (refcount=1, is_ref=0)=42,
'name' => (refcount=2, is_ref=0)='life'
)

如果將數(shù)組的引用賦值給數(shù)組中的一個(gè)元素,有意思的事情就會(huì)發(fā)生:

?php
  $a = array('one');
  $a[] = $a;
?>

這樣 a 數(shù) 組 就 有 兩 個(gè) 元 素 , 一 個(gè) 索 引 為 0 , 值 為 字 符 o n e , 另 外 一 個(gè) 索 引 為 1 , 為 a數(shù)組就有兩個(gè)元素,一個(gè)索引為0,值為字符one,另外一個(gè)索引為1,為 a數(shù)組就有兩個(gè)元素,一個(gè)索引為0,值為字符one,另外一個(gè)索引為1,為a自身的引用,內(nèi)部存儲(chǔ)如下:

a: (refcount=2, is_ref=1)=array (
0 => (refcount=1, is_ref=0)='one',
1 => (refcount=2, is_ref=1)=…
)

array這個(gè)zvalref_count是2 , 是一個(gè)環(huán)形引用
這時(shí)對$a進(jìn)行unset,那么 a 會(huì) 從 符 號(hào) 表 中 刪 除 , 同 時(shí) ‘ a會(huì)從符號(hào)表中刪除,同時(shí)` a會(huì)從符號(hào)表中刪除,同時(shí)‘a(chǎn)指向的zvalrefcount_gc`減少1.

?php
$a = array('one');
$a[] = $a;
unset($a);
?>

那么問題就產(chǎn)生了, a 已 經(jīng) 不 在 符 號(hào) 表 中 , 用 戶 無 法 再 訪 問 此 變 量 , 但 是 a已經(jīng)不在符號(hào)表中,用戶無法再訪問此變量,但是 a已經(jīng)不在符號(hào)表中,用戶無法再訪問此變量,但是a之前指向的zval的refcount_gc變?yōu)?而不是0,因此不能被回收,從而產(chǎn)生內(nèi)存泄露,新的GC要做的工作就是清理此類垃圾。

為了解決循環(huán)引用內(nèi)存泄露問題 , 使用同步周期回收算法 , 這種ref_count減1后還大于0的會(huì)被作為疑似垃圾

比如當(dāng)數(shù)組或?qū)ο笱h(huán)的引用自身 , unset掉數(shù)組的時(shí)候 , 當(dāng)refcount-1后還大于0的 , 會(huì)進(jìn)行遍歷 ,并且模擬的刪除一次refcount-1如果是0就刪除 ,如果不是0就恢復(fù)。

到此這篇關(guān)于PHP的垃圾回收機(jī)制代碼實(shí)例講解的文章就介紹到這了,更多相關(guān)PHP的垃圾回收機(jī)制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • PHP進(jìn)階學(xué)習(xí)之垃圾回收機(jī)制詳解
  • PHP析構(gòu)函數(shù)destruct與垃圾回收機(jī)制的講解
  • 簡單談?wù)凱HP的垃圾回收機(jī)制
  • 解讀PHP中的垃圾回收機(jī)制
  • PHP垃圾回收機(jī)制講解

標(biāo)簽:白城 鶴崗 克拉瑪依 遼陽 唐山 六安 鷹潭 柳州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP的垃圾回收機(jī)制代碼實(shí)例講解》,本文關(guān)鍵詞  PHP,的,垃圾,回收,機(jī)制,代碼,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《PHP的垃圾回收機(jī)制代碼實(shí)例講解》相關(guān)的同類信息!
  • 本頁收集關(guān)于PHP的垃圾回收機(jī)制代碼實(shí)例講解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章