PHP的垃圾回收機(jī)制代碼實(shí)例講解
PHP可以自動(dòng)進(jìn)行內(nèi)存管理,清除不需要的對(duì)象,主要使用了引用計(jì)數(shù)
在zval
結(jié)構(gòu)體中定義了ref_count
和is_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對(duì)應(yīng)的zval的refcount_gc
會(huì)加1
<?php $a = "new string"; $b = $a; ?>
此時(shí) a 和 a和 a和b變量對(duì)應(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' ?>
對(duì)于普通的變量來說,這一切很正常,但是在復(fù)合類型變量(數(shù)組和對(duì)象)中,會(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è)zval
的ref_count
是2 , 是一個(gè)環(huán)形引用
這時(shí)對(duì)$a
進(jìn)行unset
,那么 a 會(huì) 從 符 號(hào) 表 中 刪 除 , 同 時(shí) ‘ a會(huì)從符號(hào)表中刪除,同時(shí)` a會(huì)從符號(hào)表中刪除,同時(shí)‘a(chǎn)指向的
zval的
refcount_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)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
ThinkPHP5.0多個(gè)文件上傳后找不到臨時(shí)文件的修改方法
這篇文章主要介紹了ThinkPHP5.0多個(gè)文件上傳后找不到臨時(shí)文件的修改方法,需要的朋友可以參考下2018-07-07php 與 nginx 的處理方式及nginx與php-fpm通信的兩種方式
這篇文章主要介紹了php 與 nginx 的兩種處理方式及nginx與php-fpm通信的兩種方式,需要的朋友可以參考下2018-09-09PHP使用第三方即時(shí)獲取物流動(dòng)態(tài)實(shí)例詳解
這篇文章主要介紹了PHP使用第三方即時(shí)獲取物流動(dòng)態(tài),非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-04-04Drupal7中常用的數(shù)據(jù)庫操作實(shí)例
Drupal 7 提供的新功能其中一個(gè)就是可以使用 Query Builder and Query Objects 查詢生成器來構(gòu)造查詢對(duì)象的能力,無需在代碼中寫原始的SQL語句,一是提高了代碼可閱讀性,二是兼容所有數(shù)據(jù)庫2014-03-03Laravel等框架模型關(guān)聯(lián)的可用性淺析
這篇文章主要給大家介紹了關(guān)于Laravel等框架模型關(guān)聯(lián)的可用性的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Laravel等框架具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12變量在 PHP7 內(nèi)部的實(shí)現(xiàn)(二)
在上篇文章給大家介紹了變量在 PHP7 內(nèi)部的實(shí)現(xiàn)(一),本篇繼續(xù)給大家介紹php7內(nèi)部實(shí)現(xiàn)相關(guān)知識(shí),感興趣的朋友通過本篇文章一起學(xué)習(xí)吧2015-12-12php寫app接口并返回json數(shù)據(jù)的實(shí)例(分享)
下面小編就為大家?guī)硪黄猵hp寫app接口并返回json數(shù)據(jù)的實(shí)例(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05apache php mysql開發(fā)環(huán)境安裝教程
這篇文章主要為大家詳細(xì)介紹了apache php mysql開發(fā)環(huán)境安裝教程,感興趣的小伙伴們可以參考一下2016-07-07php中計(jì)算未知長度的字符串哪個(gè)字符出現(xiàn)的次數(shù)最多的代碼
php中計(jì)算未知長度的字符串哪個(gè)字符出現(xiàn)的次數(shù)最多的代碼,需要的朋友可以參考下2012-08-08