亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

分析PHP的垃圾回收機制

瀏覽:2日期:2022-09-07 08:35:01

如果用過C語言,那么申請內(nèi)存的方式是malloc或者是calloc,然后你用完這個內(nèi)存后,一定不要忘了用free函數(shù)去釋放掉,這就是傳說中手動垃圾回收,一般都是掃地神僧用這種方式。很多高層次語言中,你這輩子都是接觸不到內(nèi)存管理的,比如世界上最好的語言php,這種語言替你管理了內(nèi)存,你就安安心心寫爛代碼即可。寫php的,你說你關(guān)心內(nèi)存,我是不怎么相信的,一定是你在裝逼。當然了,如果你用的swoole或者wm或者自己發(fā)明的常駐內(nèi)存級php應(yīng)用,那你將不得不關(guān)注內(nèi)存泄露問題,也就說一定要記得釋放無用變量。那么,在用的最普遍地最傳統(tǒng)的web開發(fā)中,php的自動垃圾回收機制是怎樣的呢?這個問題我們先這么想,就是都知道php是C語言實現(xiàn)的,現(xiàn)在把C語言給你放在這里了,然后你想想如何用C語言實現(xiàn)對一個變量的統(tǒng)計以及釋放。你不要想如何實現(xiàn)php,你就想C語言如何實現(xiàn)一個變量,從聲明開始到最后沒人用了,就把這個變量所占的內(nèi)存給釋放掉。你從這個角度出發(fā),就會舒服一些,這不再是一個技術(shù)難題,而是一個傻逼產(chǎn)品經(jīng)理提的一個傻逼需求。好了,步入正題,PHP進行內(nèi)存管理的核心算法一共兩項:一是引用計數(shù),二是寫時拷貝,請理(bei)解(song)。當你聲明一個PHP變量的時候,C語言就在底層給你搞了一個叫做zval的struct(結(jié)構(gòu)體);如果你還給這個變量賦值了,比如“hello world”,那么C語言就在底層再給你搞一個叫做zend_value的union(聯(lián)合體),總體看來就是這樣的:

好了,進入代碼實戰(zhàn)階段,注意兩點:

$a = ’hello’. mt_rand( 1, 1000 );echo xdebug_debug_zval( ’a’);$b = $a;echo xdebug_debug_zval( ’a’);$c = $a;echo xdebug_debug_zval( ’a’);unset( $c );echo xdebug_debug_zval( ’a’);

輸出的結(jié)果是:

分析PHP的垃圾回收機制

其中,zval struct結(jié)構(gòu)體用于保存$a,zend_value union聯(lián)合體用于保存數(shù)據(jù)內(nèi)容也就是’hello916’。由于后面又聲明了b和c,所以C不得不又在底層給你搞出兩個zval struct結(jié)構(gòu)體來。

其中,zval和zend value的結(jié)構(gòu)大概如下:(注意!!!這并不是完整正確的PHP zval和zend_value在C語言中struct和union實現(xiàn),僅僅是挑出最重點的部分寫出來,強調(diào)一下:你沒有必要一個字不差背誦過zval和zend_value,你只需要知道原理)

zval {

string 'a' //變量的名字是a

value zend_value //變量的值

type string //變量是字符串類型

}

zend_value {

string 'hello916' //值的內(nèi)容

refcount 1 //引用計數(shù)

}

看到上面兩個,如果面試官問你php變量為什么能夠保存字符串'123'也能保存數(shù)字123,你知道該怎么回答了吧?就答出重點zval中有該變量的類型,當是字符串123的時候,type就是string,此時value指向“123”;當是整數(shù)123的時候,zval的type為int,value為123。這就是答題的思想,這很重要!而且,通過C語言都是可以實現(xiàn)的!具體真正的val和zend_value的模樣,有興趣的同學(xué)可以去網(wǎng)上搜搜,如果你沒有C語言的底子,可能比較吃力!前者是一個struct結(jié)構(gòu)體,后者是一個union聯(lián)合體!

這個refcount就是傳說中的引用計數(shù)了,初始化的時候a后面的引用次數(shù)為1(注意,正確說法應(yīng)該是a后面的賦值的數(shù)組zend_value引用計數(shù)為1,而不是a這個變量zval本身)。然后我們將$b = $a,其實相當于又一個變量指向了這個zend_value,所以refcount變?yōu)?,最后將$c = $a,同理,zend_value的refcount再次加1變成了3。然后,我們用unset( $c ),這會兒,C語言要做的就是把$c的zval給KO free掉,但是并不是free zend_value,這會兒zend_value的refcount就自然而然減1變成2了。

那么寫時拷貝是什么意思呢?看下面代碼:

<?php// 先不要問為什么非要加mt_rand,不然,絕筆說不過來了,到處都是坑$a = ’hello’. mt_rand( 1, 1000 );$b = $a;$a = 123;echo $b. PHP_EOL;

// 運行結(jié)果,不用我說吧,腳趾頭都知道是’hello’.mt_rand( 1, 1000 )的結(jié)果,絕對不可能是123。

其實,當你把$a賦值給$b的時候,$a的值并沒有真的復(fù)制了一份,這樣是對內(nèi)存的極度不尊重,也是對時間復(fù)雜度的極度不尊重,計算機僅僅是將$b指向了$a的值而已,這就叫多快好省。那么,什么時候真正的發(fā)生復(fù)制呢?就是當我們修改$a的值為123的時候,這個時候就不得已進行復(fù)制,避免$b的值和$a的一樣。

<?php$a = ’hello’. mt_rand( 1, 1000 );$b = $a;echo xdebug_debug_zval( ’a’);$a = ’world’. mt_rand( 2, 2000 );echo xdebug_debug_zval( ’a’);

// 運行結(jié)果為1,其中的原理你自己應(yīng)該能理順了昂

叨逼叨了這么長,通過簡單的案例解釋清楚了兩個要點:引用計數(shù)和寫時拷貝,那么垃圾回收也該來了。當一個zval在被unset的時候、或者從一個函數(shù)中運行完畢出來(就是局部變量)的時候等等很多地方,都會產(chǎn)生zval與zend_value發(fā)生斷開的行為,這個時候zend引擎需要檢測的就是zend_value的refcount是否為0,如果為0,則直接KO free空出內(nèi)容來。如果zend_value的recount不為0(廢話一定是大于0),這個value不能被釋放,但是也不代表這個zend_value是清白的,因為此zend_value依然可能是個垃圾。

什么樣的情況會導(dǎo)致zend_value的refcount不為0,但是這個zend_value卻是個垃圾呢?PHP7種兩種情況:

<?php$arr = [ 1 ];$arr[] = &$arr;unset( $arr );

這種情況下,zend_value不會能釋放,但也不能放過它,不然一定會產(chǎn)生內(nèi)存泄漏,所以這會兒zend_value會被扔到一個叫做垃圾回收堆中,然后zend引擎會依次對垃圾回收堆中的這些zend_value進行二次檢測,檢測是不是由于上述兩種情況造成的refcount為1但是自身卻確實沒有人再用了,如果一旦確定是上述兩種情況造成的,那么就會將zend_value徹底抹掉釋放內(nèi)存。

那么垃圾回收發(fā)生在什么時候?有些同學(xué)可能有疑問,就是php不是運行一次就銷毀了嗎,我要著gc有何用?并不是啦,首先當一次fpm運行完畢后,最后一定還有g(shù)c的,這個銷毀就是gc;其次是,內(nèi)存都是即用即釋放的,而不是攢著非得到最后,你想想一個典型的場景,你的控制器里的某個方法里用了一個函數(shù),函數(shù)需要一個巨大的數(shù)組參數(shù),然后函數(shù)還需要修改這個巨大的數(shù)組參數(shù),你們應(yīng)該是函數(shù)的運行范圍里面修改這個數(shù)組,所以此時會發(fā)生寫時拷貝了,當函數(shù)運行完畢后,就得趕緊釋放掉這塊兒內(nèi)存以供給其他進程使用,而不是非得等到本地fpm request徹底完成后才銷毀。

說到最后,說些自己的話:大多數(shù)情況下,面試官問你問題主要是想一是要你個思維思路,二是看你學(xué)習(xí)程度。就像gc這個問題,其實很多腳本語言的垃圾回收機制基本上都是靠引用計數(shù)和寫時拷貝這兩種算法結(jié)合完成的,所以如果你設(shè)計一門腳本語言,gc機制就按照這兩種算法進行設(shè)計即可。其次是大多數(shù)phper不會看這些東西的,面試官問你這個問題不是要你死記硬背那么多細節(jié),你背不過的,他還是想探測你平時有沒有更積極地往深層發(fā)展的心態(tài)。

注重體現(xiàn)重點,很多細節(jié)實在沒法寫,比如我舉個例子$a=[],xdebug_debug_zval( $a )的refcount值你猜是多少? 7.1.17下竟然是2,你是不是以為是1,然而并不是。不過你不用糾結(jié)這些細節(jié),gc的關(guān)鍵就是能說出引用計數(shù)的原理和寫時拷貝,很多細節(jié)深處都各種奇奇怪怪的東西,面試官自己都不一定知道。

以上就是淺談PHP的垃圾回收機制的詳細內(nèi)容,更多關(guān)于PHP的垃圾回收機制的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標簽: PHP
相關(guān)文章:
主站蜘蛛池模板: aaaaa毛片| 欧美日韩视频在线一区二区 | 欧美aaa大片| 黄色三级国产 | 黄色福利在线观看 | 国产精品91在线播放 | 97国产超级碰碰在线视频 | 九九精品在线播放 | 国产成人精品免费午夜 | 久久月| 97视频精品全国在线观看 | 国产午夜精品视频 | 欧美日韩视频一区二区 | 国产丝袜按摩女技师在线 | 天天干天天色综合网 | 黑人的逼 | 一级做a爰视频免费观看2019 | 一级黄色片免费的 | 亚洲国产99在线精品一区二区 | 国产成人短视频在线观看免费 | 国产乱子精品免费视观看片 | 欧美黄色大片免费看 | 亚洲精品国产精品乱码不97 | 亚洲国产精品专区 | 亚洲精品美女一区二区三区乱码 | 成人国产欧美精品一区二区 | 欧美在线黄色 | 国产小毛片 | 99久久精品国产免费 | 欧美在线 | 亚洲 | 青草悠悠视频在线观看 | 大学生一级特黄的免费大片视频 | 亚洲欧美一 | 亚洲另类欧美日韩 | 免费人成黄页在线观看忧物 | 被黑人做的白浆直流在线播放 | 久草一级片 | 911亚洲精品 | 午夜黄色毛片 | 国产美女久久久 | 亚洲欧美国产另类 |