解讀PHP中的垃圾回收機(jī)制
PHP的基本GC概念
PHP語(yǔ)言同其他語(yǔ)言一樣,具有垃圾回收機(jī)制。那么今天我們要為大家講解的內(nèi)容就是關(guān)于PHP垃圾回收機(jī)制的相關(guān)問(wèn)題。希望對(duì)大家有所幫助。PHP strtotime應(yīng)用經(jīng)驗(yàn)之談PHP memory_get_usage()管理內(nèi)存PHP unset全局變量運(yùn)用問(wèn)題詳解PHP unset()函數(shù)銷(xiāo)毀變量教你快速實(shí)現(xiàn)PHP全站權(quán)限驗(yàn)證一、PHP 垃圾回收機(jī)制(Garbage Collector 簡(jiǎn)稱(chēng)GC) 在PHP中,沒(méi)有任何變量指向這個(gè)對(duì)象時(shí),這個(gè)對(duì)象就成為垃圾。PHP會(huì)將其在內(nèi)存中銷(xiāo)毀;這是PHP的GC垃圾處理機(jī)制,防止內(nèi)存溢出。當(dāng)一個(gè)PHP線程結(jié)束時(shí),當(dāng)前占用的所有內(nèi)存空間都會(huì)被銷(xiāo)毀,當(dāng)前程序中所有對(duì)象同時(shí)被銷(xiāo)毀。GC進(jìn)程一般都跟著每起一個(gè)SESSION而開(kāi)始運(yùn)行的.gc目的是為了在session文件過(guò)期以后自動(dòng)銷(xiāo)毀刪除這些文件.二、__destruct /unset __destruct() 析構(gòu)函數(shù),是在垃圾對(duì)象被回收時(shí)執(zhí)行。
unset 銷(xiāo)毀的是指向?qū)ο蟮淖兞浚皇沁@個(gè)對(duì)象。三、 Session 與PHP垃圾回收機(jī)制由于PHP的工作機(jī)制,它并沒(méi)有一個(gè)daemon線程來(lái)定期的掃描Session信息并判斷其是否失效,當(dāng)一個(gè)有效的請(qǐng)求發(fā)生時(shí),PHP 會(huì)根據(jù)全局變量 session.gc_probability和session.gc_divisor的值,來(lái)決定是否啟用一個(gè)GC, 在默認(rèn)情況下,session.gc_probability=1, session.gc_divisor =100也就是說(shuō)有1%的可能性啟動(dòng)GC(也就是說(shuō)100個(gè)請(qǐng)求中只有一個(gè)gc會(huì)伴隨100個(gè)中的某個(gè)請(qǐng)求而啟動(dòng)).PHP垃圾回收機(jī)制的工作就是掃描所有的Session信息,用當(dāng)前時(shí)間減去session最后修改的時(shí)間,同session.gc_maxlifetime參數(shù)進(jìn)行比較,如果生存時(shí)間超過(guò)gc_maxlifetime(默認(rèn)24分鐘),就將該session刪除。
但是,如果你Web服務(wù)器有多個(gè)站點(diǎn),多個(gè)站點(diǎn)時(shí),GC處理session可能會(huì)出現(xiàn)意想不到的結(jié)果,原因就是:GC在工作時(shí),并不會(huì)區(qū)分不同站點(diǎn)的session.那么這個(gè)時(shí)候怎么解決呢?
1. 修改session.save_path,或使用session_save_path()讓每個(gè)站點(diǎn)的session保存到一個(gè)專(zhuān)用目錄,
2. 提供GC的啟動(dòng)率,自然,PHP垃圾回收機(jī)制的啟動(dòng)率提高,系統(tǒng)的性能也會(huì)相應(yīng)減低,不推薦。
3. 在代碼中判斷當(dāng)前session的生存時(shí)間,利用session_destroy()刪除。
引用計(jì)數(shù)基本知識(shí)
每個(gè)php變量存在一個(gè)叫做"zval"的變量容器中.一個(gè)zval變量容器,除了包含變量的類(lèi)型和值,還包括兩個(gè)字節(jié)的額外信息.第一個(gè)是"is_ref",是個(gè)bool值,用來(lái)標(biāo)識(shí)這個(gè)變量是否是屬于引用集合(reference set).通過(guò)這個(gè)字節(jié),php引擎才能把普通變量和引用變量區(qū)分開(kāi).由于php允許用戶(hù)通過(guò)使用&來(lái)使用自定義引用,zval變量容器中還有一個(gè)內(nèi)部引用計(jì)數(shù)機(jī)制,來(lái)優(yōu)化內(nèi)存使用.第二個(gè)額外字節(jié)是"refcount",用來(lái)表示指向這個(gè)zval變量容器的變量(也稱(chēng)符號(hào)即symbol)個(gè)數(shù).
當(dāng)一個(gè)變量被賦常量值時(shí),就會(huì)生成一個(gè)zval變量容器,如下例所示:
<?php $a = "new string"; ?>
在上例中,新的變量是a,是在當(dāng)前作用域中生成的.并且生成了類(lèi)型為string和值為"new string"的變量容器.在額外的兩個(gè)字節(jié)信息中,"is_ref"被默認(rèn)設(shè)置為false,因?yàn)闆](méi)有任何自定義的引用生成."refcount"被設(shè)定為1,因?yàn)檫@里只有一個(gè)變量使用這個(gè)變量容器.調(diào)用xdebug查看一下變量?jī)?nèi)容:
<?php $a = "new string"; xdebug_debug_zval('a'); ?>
以上代碼會(huì)輸出:
a: (refcount=1, is_ref=0)='new string'
對(duì)變量a增加一個(gè)引用計(jì)數(shù)
<?php $a = "new string"; $b = $a; xdebug_debug_zval('a'); ?>
以上代碼會(huì)輸出:
a: (refcount=2, is_ref=0)='new string'
這時(shí),引用次數(shù)是2,因?yàn)橥蛔兞咳萜鞅蛔兞縜和變量b關(guān)聯(lián).當(dāng)沒(méi)必要時(shí),php不會(huì)去復(fù)制已生成的變量容器.變量容器在"refcount"變成0時(shí)就被銷(xiāo)毀.當(dāng)任何關(guān)聯(lián)到某個(gè)變量容易的變量離開(kāi)它的作用域(比如:函數(shù)執(zhí)行結(jié)束),或者對(duì)變量調(diào)用了unset()函數(shù),"refcount"就會(huì)減1,下面例子就能說(shuō)明:
<?php $a = "new string"; $b = $c = $a; xdebug_debug_zval('a'); unset($b, $c); xdebug_debug_zval('a'); ?>
以上代碼會(huì)輸出:
a: (refcount=3, is_ref=0)='new string' a: (refcount=1, is_ref=0)='new string'
如果我們現(xiàn)在執(zhí)行unset($a),$包含的類(lèi)型和值的這個(gè)容器就會(huì)從內(nèi)存刪除
復(fù)合類(lèi)型(compound types)
當(dāng)考慮像array和object這樣的復(fù)合類(lèi)型時(shí),事情會(huì)稍微有些復(fù)雜.與標(biāo)量(scalar)類(lèi)型的值不同,array和object類(lèi)型的變量把它們的成員或?qū)傩源嬖谧约旱姆?hào)表中.這意味著下面的例子將生成三個(gè)zval變量容器
<?php $a = array('meaning' => 'life', 'number' => 42); xdebug_debug_zval('a'); ?>
以上代碼輸出:
a: (refcount=1, is_ref=0)=array ('meaning' => (refcount=1, is_ref=0)='life', 'number' => (refcount=1, is_ref=0)=42)
這三個(gè)zval變量容器是:a,meaning,number.增加和減少refcount的規(guī)則和上面提到的一樣
特例,添加數(shù)組本身作為數(shù)組元素時(shí):
<?php $a = array('one'); $a[] = &$a; xdebug_debug_zval('a'); ?>
以上代碼輸出的結(jié)果:
a: (refcount=2, is_ref=1)=array (0 => (refcount=1, is_ref=0)='one', 1 => (refcount=2, is_ref=1)=...)
可以看到數(shù)組a和數(shù)組本身元素a[1]指向的變量容器refcount為2
當(dāng)對(duì)數(shù)組$a調(diào)用unset函數(shù)時(shí),$a的refcount變?yōu)?,發(fā)生了內(nèi)存泄漏
清理變量容器的問(wèn)題
盡管不再有某個(gè)作用域中的任何符號(hào)指向這個(gè)結(jié)構(gòu)(就是變量容器),由于數(shù)組元素"1"仍然指向數(shù)組本身,所以這個(gè)容器不能被消除.因?yàn)闆](méi)有另外的符號(hào)指向它,用戶(hù)沒(méi)有辦法清除這個(gè)結(jié)構(gòu),結(jié)果就會(huì)導(dǎo)致內(nèi)存泄漏.慶幸的是,php將在請(qǐng)求結(jié)束時(shí)清除這個(gè)數(shù)據(jù)結(jié)構(gòu),但是php清除前,將耗費(fèi)不少內(nèi)存空間
回收周期
5.3.0PHP使用了新的同步周期回收算法,來(lái)處理上面所說(shuō)的內(nèi)存泄漏問(wèn)題
首先,我們先要建立一些基本規(guī)則:
如果一個(gè)引用計(jì)數(shù)增加,它將繼續(xù)被使用,當(dāng)然就不再垃圾中.如果引用技術(shù)減少到零,所在的變量容器將被清除(free).就是說(shuō),僅僅在引用計(jì)數(shù)減少到非零值時(shí),才會(huì)產(chǎn)生垃圾周期(grabage cycle).其次,在一個(gè)垃圾周期中,通過(guò)檢查引用計(jì)數(shù)是否減1,并且檢查哪些變量容器的引用次數(shù)是零,來(lái)發(fā)現(xiàn)哪部分是垃圾
為避免不得不檢查所有引用計(jì)數(shù)可能減少的垃圾周期,這個(gè)算法把所有可能根(possible roots 都是zval變量容器),放在根緩沖區(qū)(root buffer)中(用紫色標(biāo)記),這樣可以同時(shí)確保每個(gè)可能的垃圾根(possible garbage root)在緩沖區(qū)只出現(xiàn)一次.僅僅在根緩沖區(qū)滿(mǎn)了時(shí),才對(duì)緩沖區(qū)內(nèi)部所有不同的變量容器執(zhí)行垃圾回收操作。
相關(guān)文章
功能強(qiáng)大的PHP POST提交數(shù)據(jù)類(lèi)
這篇文章主要為大家詳細(xì)介紹了功能強(qiáng)大的PHP POST提交數(shù)據(jù)類(lèi),代碼簡(jiǎn)潔且具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07PHP stripos()函數(shù)及注意事項(xiàng)的分析
本篇文章是對(duì)PHP中的stripos()函數(shù)及注意事項(xiàng)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06PHP最常用的2種設(shè)計(jì)模式工廠模式和單例模式介紹
當(dāng)你在不斷的試圖從你的應(yīng)用程序中發(fā)現(xiàn)新的特征時(shí),你是否發(fā)現(xiàn)你提出的解決方法和一些以前你已經(jīng)實(shí)現(xiàn)的東西是如此的類(lèi)似呢2012-08-08php使用filter_var函數(shù)判斷郵箱,url,ip格式示例
這篇文章主要介紹了php使用filter_var函數(shù)判斷郵箱,url,ip格式,簡(jiǎn)單分析了php filter_var函數(shù)的功能、參數(shù),并結(jié)合實(shí)例形式給出了filter_var函數(shù)判斷郵箱,url,ip格式的相關(guān)操作技巧,需要的朋友可以參考下2019-07-07PHP pathinfo()獲得文件的路徑、名稱(chēng)等信息說(shuō)明
在PHP中,若想通過(guò)函數(shù)獲得一個(gè)文件的路徑、名稱(chēng),或者是擴(kuò)展名等,是非常容易的一件事??梢允褂胐irname()、basename()、pathinfo()等多種途徑獲得相應(yīng)的信息。2011-09-09