這篇文章將為大家詳細(xì)講解有關(guān)PHP垃圾回收機(jī)制的案例分析,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。
創(chuàng)新互聯(lián)于2013年開(kāi)始,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站建設(shè)、網(wǎng)站制作網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元周村做網(wǎng)站,已為上家服務(wù),為周村各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:028-86922220PHP垃圾回收機(jī)制:
1. PHP可以自動(dòng)進(jìn)行內(nèi)存管理,清除不需要的對(duì)象,主要使用了引用計(jì)數(shù)
2. 在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)制引用
3. 為了解決循環(huán)引用內(nèi)存泄露問(wèn)題 , 使用同步周期回收算法。
比如當(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)生過(guò)程:
<?php $a = "new string"; ?>
a: (refcount_gc=1, is_ref_gc=0)='new string'
當(dāng)把$a賦值給另外一個(gè)變量的時(shí)候,$a對(duì)應(yīng)的zval的refcount_gc會(huì)加1
<?php $a = "new string"; $b = $a; ?>
此時(shí)$a和$b變量對(duì)應(yīng)的內(nèi)部存儲(chǔ)信息為,$a和$b同時(shí)指向一個(gè)字符串"new string" ,它的refcount變成2a,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ì)于普通的變量來(lái)說(shuō),這一切很正常,但是在復(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ù)原則,用圖來(lái)表示:
下面在$a中添加一個(gè)元素,并將現(xiàn)有的一個(gè)元素的值賦給新的元素:
<?php $a = array('meaning' => 'life', 'number' => 42); $a['name'] = $a['meaning']; ?>
那么$a的內(nèi)部存儲(chǔ)為 , "life" 的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,值為字符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指向的zval的refcount_gc減少1.
那么問(wèn)題就產(chǎn)生了,$a已經(jīng)不在符號(hào)表中,用戶無(wú)法再訪問(wèn)此變量,但是$a之前指向的zval的refcount_gc變?yōu)?而不是0,因此不能被回收,從而產(chǎn)生內(nèi)存泄露,新的GC要做的工作就是清理此類垃圾。
為了解決循環(huán)引用內(nèi)存泄露問(wèn)題 , 使用同步周期回收算法 , 這種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ī)制的案例分析就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。
文章標(biāo)題:PHP垃圾回收機(jī)制的案例分析-創(chuàng)新互聯(lián)
當(dāng)前URL:http://www.rwnh.cn/article22/hsgjc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站內(nèi)鏈、定制開(kāi)發(fā)、動(dòng)態(tài)網(wǎng)站、定制網(wǎng)站、虛擬主機(jī)、自適應(yīng)網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容