摘要:引入的同步算法傳統(tǒng)上,像以前的用到的引用計數(shù)內(nèi)存機(jī)制,無法處理循環(huán)引用的內(nèi)存泄漏。然而使用文章引用計數(shù)系統(tǒng)中的同步周期回收中的同步算法,解決了這個內(nèi)存泄漏問題,這種算法就是的垃圾回收機(jī)制。
引用賦值
$a = "apple"; $b = &$a;
上述代碼中,我將一個字符串賦值給變量a,然后將a的引用賦值給了變量b。顯然,這個時候的內(nèi)存指向應(yīng)該是這樣的:
$a -> "apple" <- $b
a和b指向了同一塊內(nèi)存區(qū)域(變量容器 zval ),我們通過 var_dump($a, $b) 得到 string(5) "apple" string(5) "apple" ,這是我們預(yù)期的結(jié)果。
unset函數(shù) 與 引用計數(shù) unset 函數(shù)假如我想將 "apple" 這個字符串從內(nèi)存中釋放掉。我是這么做的:
unset($a);
但是通過再次打印 $a $b 兩變量的信息,我得到了這樣的結(jié)果:Notice: Undefined variable: a 和 string(5) "apple" 。奇怪,$a $b 指向同一個變量容器,又明明將$a釋放了,為什么$b還是"apple"。
其實(shí)是這樣的,unset()只是將一個變量符號a(指針)銷毀了,并沒有釋放掉那個變量容器,所以執(zhí)行完操作之后,內(nèi)存指向只是變成了這樣:
"apple" <- $b引用計數(shù)
引用計數(shù) (reference count)是每個變量容器中都會存放的一條信息,它表示當(dāng)前變量容器正被多少個變量符號所引用。
正如之前的例子,unset()并沒有釋放變量所指向的變量容器,而只是將變量符號銷毀了。同時,將變量容器中的 引用計數(shù) 減1,當(dāng)引用計數(shù)為0時,也就是說當(dāng)變量容器不被任何變量引用時,便會觸發(fā)php的垃圾回收(錯誤) ,它便會被釋放(正確)。
更正上述的一個小錯誤: 這種單純的引用計數(shù)方式是 php 5.2 之前的內(nèi)存管理機(jī)制,稱不上是垃圾回收機(jī)制,垃圾回收機(jī)制是 php 5.3 才引入的,垃圾回收機(jī)制為的是解決這種單純的引用計數(shù)內(nèi)存管理機(jī)制的缺陷(即 循環(huán)引用導(dǎo)致的內(nèi)存泄漏,下文會進(jìn)行講解)
回到正題,我們用代碼來驗證一下先前的結(jié)論:
$a = "apple"; $b = &$a; $before = memory_get_usage(); unset($a); $after = memory_get_usage(); var_dump($before - $after); // 結(jié)果為int(0),變量容器的引用計數(shù)為1,沒有釋放
$a = "apple"; $b = &$a; $before = memory_get_usage(); unset($a, $b); $after = memory_get_usage(); var_dump($before - $after); // 結(jié)果為int(24),變量容器的引用計數(shù)為0,得到釋放直接釋放
那要怎樣做才能真正釋放掉 "apple" 所占用的內(nèi)存呢?
利用上述方法,我們可以在 unset($a) 之后再 unset($b) ,將變量容器的所有引用都銷毀,引用計數(shù)減為0了,自然就被釋放掉了。
當(dāng)然,還有更直接的方法:
$a = null;
直接賦值 null 會將 $a 所指向的內(nèi)存區(qū)域置空,并將引用計數(shù)歸零,內(nèi)存便被釋放。
腳本執(zhí)行結(jié)束后的內(nèi)存對于一般的web程序來說(fpm模式下),php的執(zhí)行是單線程同步阻塞型的,當(dāng)腳本執(zhí)行結(jié)束之后,腳本內(nèi)使用的所有內(nèi)存都會被釋放。那么,我們手動去釋放內(nèi)存到底有意義嗎?
其實(shí)關(guān)于這個問題,早有解答,推薦大家看一下鳥哥 @laruence 2012年發(fā)表的一篇文章:
請手動釋放你的資源(Please release resources manually)引用計數(shù)內(nèi)存管理機(jī)制的缺陷:循環(huán)引用
現(xiàn)在我們來講講之前提到的引用計數(shù)內(nèi)存管理機(jī)制的缺陷。
當(dāng)一個變量容器的引用計數(shù)為0時,php會進(jìn)行垃圾回收。但是,你可想過,有一種情況會導(dǎo)致一個變量容器的引用計數(shù)永遠(yuǎn)不會被減為0,舉個例子:
$a = ["one"]; $a[] = &$a;
我們看到,$a數(shù)組第二個元素就是它本身。那么,存放數(shù)組的這個變量容器的引用計數(shù)為2,一個引用是變量a,另一個引用是這個數(shù)組的第二個元素 - 索引1。
那么,如果這時我們 unset($a) ,存放數(shù)組的變量容器的引用計數(shù)會減1,但還有1個引用,就是數(shù)組的元素 1 ,現(xiàn)在引用結(jié)構(gòu)變成了這樣:
由于變量容器的引用計數(shù)沒有變?yōu)?,所以不能被釋放,而且這時又沒有外部其他變量符號引用它,用戶也沒有辦法去清除這個結(jié)構(gòu),這時它就會一直駐留在內(nèi)存之中。
所以如果代碼中存在大量的這種結(jié)構(gòu)和操作,最終會導(dǎo)致內(nèi)存損耗甚至泄漏。這就是 循環(huán)引用 帶來的內(nèi)存無法釋放的問題。
慶幸的是,fpm模式下,當(dāng)請求的腳本執(zhí)行結(jié)束,php會釋放所有腳本中使用到的內(nèi)存,包括這個結(jié)構(gòu)。但是,如果是守護(hù)進(jìn)程下的php程序呢?比如swoole。這個php需要解決的急迫問題(已經(jīng)解決,見下文)。
傳統(tǒng)上,像以前的 php 用到的引用計數(shù)內(nèi)存機(jī)制,無法處理循環(huán)引用的內(nèi)存泄漏。然而 5.3.0 PHP 使用文章 ? 引用計數(shù)系統(tǒng)中的同步周期回收(Concurrent Cycle Collection in Reference Counted Systems) 中的同步算法,解決了這個內(nèi)存泄漏問題,這種算法就是PHP的垃圾回收機(jī)制。
具體算法的實(shí)現(xiàn)和流程有些許復(fù)雜,請閱讀官方文檔,這里不再贅述,另附上幾個算法流程講解的文章鏈接,講得比較直白:
http://php.net/manual/zh/feat... 官方文檔
http://www.cnblogs.com/leoo2s...
https://blog.csdn.net/phpkern...
最后,還是引用鳥哥文章的這兩段來說明問題:
在PHP5.2以前, PHP使用引用計數(shù)(Reference count)來做資源管理, 當(dāng)一個zval的引用計數(shù)為0的時候, 它就會被釋放. 雖然存在循環(huán)引用(Cycle reference), 但這樣的設(shè)計對于開發(fā)Web腳本來說, 沒什么問題, 因為Web腳本的特點(diǎn)和它追求的目標(biāo)就是執(zhí)行時間短, 不會長期運(yùn)行. 對于循環(huán)引用造成的資源泄露, 會在請求結(jié)束時釋放掉. 也就是說, 請求結(jié)束時釋放資源, 是一種補(bǔ)救措施(backup).然而, 隨著PHP被越來越多的人使用, 就有很多人在一些后臺腳本使用PHP, 這些腳本的特點(diǎn)是長期運(yùn)行, 如果存在循環(huán)引用, 導(dǎo)致引用計數(shù)無法及時釋放不用的資源, 則這個腳本最終會內(nèi)存耗盡退出.
所以在PHP5.3以后, 我們引入了GC, 也就是說, 我們引入GC是為了解決用戶無法解決的問題.
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/28499.html
摘要:總結(jié)垃圾回收機(jī)制以的引用計數(shù)機(jī)制為基礎(chǔ)以前只有該機(jī)制同時使用根緩沖區(qū)機(jī)制,當(dāng)發(fā)現(xiàn)有存在循環(huán)引用的時,就會把其投入到根緩沖區(qū),當(dāng)根緩沖區(qū)達(dá)到配置文件中的指定數(shù)量后,就會進(jìn)行垃圾回收,以此解決循環(huán)引用導(dǎo)致的內(nèi)存泄漏問題開始引入該機(jī)制 php垃圾回收機(jī)制,對于PHPer來說是一個不陌生但是又不是很熟悉的內(nèi)容。那么php是怎么實(shí)現(xiàn)對不需要的內(nèi)存進(jìn)行回收的呢? php變量的內(nèi)部存儲結(jié)構(gòu) 首先還是...
摘要:一內(nèi)存生命周期分配需要的內(nèi)存初始化值時使用分配的內(nèi)存不需要時將其內(nèi)存釋放垃圾回收器注意全局變量的生命周期直至瀏覽器卸載頁面才會結(jié)束。 一、內(nèi)存生命周期 1、分配需要的內(nèi)存(初始化值時)2、使用分配的內(nèi)存3、不需要時將其內(nèi)存釋放(垃圾回收器)注意:(1)全局變量的生命周期直至瀏覽器卸載頁面才會結(jié)束。(2)局部變量只在函數(shù)的執(zhí)行過程中存在,而在這個過程中會為局部變量在棧或堆上分配相應(yīng)的空間...
摘要:內(nèi)存回收此時,局部變量就沒有存在的必要了,因此可以釋放它們的內(nèi)存以供將來使用。局部變量會在它們離開執(zhí)行環(huán)境時自動被解除引用,如下面這個例子所示手工解除的引用由于局部變量在函數(shù)執(zhí)行完畢后就離開了其執(zhí)行環(huán)境,因此無需我們顯式地去為它解除引用。 JavaScript 具有自動垃圾收集機(jī)制(GC:Garbage Collecation),也就是說,執(zhí)行環(huán)境會負(fù)責(zé)管理代碼執(zhí)行過程中使用的內(nèi)存。而...
摘要:原文回收周期,增加一篇論文在底部。就是說,僅僅在引用計數(shù)減少到非零值時,才會產(chǎn)生垃圾周期。如果他們是循環(huán)引用周期的一部分,將永不能被清除進(jìn)而導(dǎo)致內(nèi)存泄漏。這個函數(shù)將返回使用這個算法回收的周期數(shù)。引用計數(shù)系統(tǒng)中的同步周期回收 原文:回收周期(Collecting Cycles) ,增加一篇論文在底部。 以下過程僅對數(shù)組和對象類型起作用。 傳統(tǒng)上,像以前的 php 用到的引用計數(shù)內(nèi)存機(jī)制,...
摘要:在中算法,當(dāng)節(jié)點(diǎn)緩沖區(qū)滿了之后,垃圾分析算法就會啟動,并且會釋放掉發(fā)現(xiàn)的垃圾,從而回收內(nèi)存。在編程中程序員不需要手動處理內(nèi)存資源分配與釋放,意味著本身實(shí)現(xiàn)了垃圾回收處理機(jī)制。 PHP是一種弱類型的腳本語言,弱類型不表示PHP變量沒有類型的區(qū)別,PHP變量有8種原始類型:四種標(biāo)量類型: boolean(布爾值) integer(整型) float(浮點(diǎn)型) 兩種復(fù)合類型: arra...
閱讀 1202·2021-11-24 09:38
閱讀 2608·2021-09-27 14:00
閱讀 1166·2019-08-30 15:55
閱讀 1345·2019-08-30 14:16
閱讀 1495·2019-08-30 10:54
閱讀 2869·2019-08-28 17:58
閱讀 762·2019-08-26 13:22
閱讀 1238·2019-08-26 12:01