《PHP實戰(zhàn):解讀PHP中的垃圾回收機(jī)制》要點:
本文介紹了PHP實戰(zhàn):解讀PHP中的垃圾回收機(jī)制,希望對您有用。如果有疑問,可以聯(lián)系我們。
PHP學(xué)習(xí)PHP的基本GC概念
PHP語言同其他語言一樣,具有垃圾回收機(jī)制.那么今天我們要為大家講解的內(nèi)容就是關(guān)于PHP垃圾回收機(jī)制的相關(guān)問題.希望對大家有所幫助.PHP strtotime應(yīng)用經(jīng)驗之談PHP memory_get_usage()管理內(nèi)存PHP unset全局變量運用問題詳解PHP unset()函數(shù)銷毀變量教你快速實現(xiàn)PHP全站權(quán)限驗證一、PHP 垃圾回收機(jī)制(Garbage Collector 簡稱GC) 在PHP中,沒有任何變量指向這個對象時,這個對象就成為垃圾.PHP會將其在內(nèi)存中銷毀;這是PHP的GC垃圾處理機(jī)制,防止內(nèi)存溢出.當(dāng)一個PHP線程結(jié)束時,當(dāng)前占用的所有內(nèi)存空間都會被銷毀,當(dāng)前程序中所有對象同時被銷毀.GC進(jìn)程一般都跟著每起一個SESSION而開始運行的.gc目的是為了在session文件過期以后自動銷毀刪除這些文件.二、__destruct /unset __destruct() 析構(gòu)函數(shù),是在垃圾對象被回收時執(zhí)行.
unset 銷毀的是指向?qū)ο蟮淖兞?而不是這個對象.三、 Session 與PHP垃圾回收機(jī)制由于PHP的工作機(jī)制,它并沒有一個daemon線程來定期的掃描Session信息并判斷其是否失效,當(dāng)一個有效的哀求發(fā)生時,PHP 會根據(jù)全局變量 session.gc_probability和session.gc_divisor的值,來決定是否啟用一個GC, 在默認(rèn)情況下,session.gc_probability=1, session.gc_divisor =100也就是說有1%的可能性啟動GC(也就是說100個哀求中只有一個gc會伴隨100個中的某個哀求而啟動).PHP垃圾回收機(jī)制的工作就是掃描所有的Session信息,用當(dāng)前時間減去session最后修改的時間,同session.gc_maxlifetime參數(shù)進(jìn)行比較,如果生存時間超過gc_maxlifetime(默認(rèn)24分鐘),就將該session刪除.
但是,如果你Web服務(wù)器有多個站點,多個站點時,GC處理session可能會出現(xiàn)意想不到的結(jié)果,原因就是:GC在工作時,并不會區(qū)分不同站點的session.那么這個時候怎么解決呢?
1. 修改session.save_path,或使用session_save_path()讓每個站點的session保存到一個專用目錄,
2. 提供GC的啟動率,自然,PHP垃圾回收機(jī)制的啟動率提高,系統(tǒng)的性能也會相應(yīng)減低,不推薦.
3. 在代碼中判斷當(dāng)前session的生存時間,利用session_destroy()刪除.
PHP學(xué)習(xí)
引用計數(shù)基本知識
每個php變量存在一個叫做"zval"的變量容器中.一個zval變量容器,除了包括變量的類型和值,還包括兩個字節(jié)的額外信息.第一個是"is_ref",是個bool值,用來標(biāo)識這個變量是否是屬于引用集合(reference set).通過這個字節(jié),php引擎才能把普通變量和引用變量區(qū)分開.由于php允許用戶通過使用&來使用自定義引用,zval變量容器中還有一個內(nèi)部引用計數(shù)機(jī)制,來優(yōu)化內(nèi)存使用.第二個額外字節(jié)是"refcount",用來表示指向這個zval變量容器的變量(也稱符號即symbol)個數(shù).
PHP學(xué)習(xí)當(dāng)一個變量被賦常量值時,就會生成一個zval變量容器,如下例所示:
PHP學(xué)習(xí)
<?php
$a = "new string";
?>
PHP學(xué)習(xí)
在上例中,新的變量是a,是在當(dāng)前作用域中生成的.并且生成了類型為string和值為"new string"的變量容器.在額外的兩個字節(jié)信息中,"is_ref"被默認(rèn)設(shè)置為false,因為沒有任何自定義的引用生成."refcount"被設(shè)定為1,因為這里只有一個變量使用這個變量容器.調(diào)用xdebug查看一下變量內(nèi)容:
PHP學(xué)習(xí)
<?php
$a = "new string";
xdebug_debug_zval('a');
?>
PHP學(xué)習(xí)
以上代碼會輸出:
PHP學(xué)習(xí)
a: (refcount=1, is_ref=0)='new string'
PHP學(xué)習(xí)
對變量a增加一個引用計數(shù)
PHP學(xué)習(xí)
<?php
$a = "new string";
$b = $a;
xdebug_debug_zval('a');
?>
PHP學(xué)習(xí)
以上代碼會輸出:
PHP學(xué)習(xí)
a: (refcount=2, is_ref=0)='new string'
PHP學(xué)習(xí)
這時,引用次數(shù)是2,因為同一變量容器被變量a和變量b關(guān)聯(lián).當(dāng)沒必要時,php不會去復(fù)制已生成的變量容器.變量容器在"refcount"變成0時就被銷毀.當(dāng)任何關(guān)聯(lián)到某個變量容易的變量離開它的作用域(好比:函數(shù)執(zhí)行結(jié)束),或者對變量調(diào)用了unset()函數(shù),"refcount"就會減1,下面例子就能說明:
PHP學(xué)習(xí)
<?php
$a = "new string";
$b = $c = $a;
xdebug_debug_zval('a');
unset($b, $c);
xdebug_debug_zval('a');
?>
PHP學(xué)習(xí)
以上代碼會輸出:
PHP學(xué)習(xí)
a: (refcount=3, is_ref=0)='new string' a: (refcount=1, is_ref=0)='new string'
PHP學(xué)習(xí)
如果我們現(xiàn)在執(zhí)行unset($a),$包括的類型和值的這個容器就會從內(nèi)存刪除
PHP學(xué)習(xí)復(fù)合類型(compound types)
PHP學(xué)習(xí)當(dāng)考慮像array和object這樣的復(fù)合類型時,事情會稍微有些復(fù)雜.與標(biāo)量(scalar)類型的值不同,array和object類型的變量把它們的成員或?qū)傩源嬖诒旧淼姆柋碇?這意味著下面的例子將生成三個zval變量容器
PHP學(xué)習(xí)
<?php
$a = array('meaning' => 'life', 'number' => 42);
xdebug_debug_zval('a');
?>
PHP學(xué)習(xí)
以上代碼輸出:
PHP學(xué)習(xí)
a: (refcount=1, is_ref=0)=array ('meaning' => (refcount=1, is_ref=0)='life', 'number' => (refcount=1, is_ref=0)=42)
PHP學(xué)習(xí)
這三個zval變量容器是:a,meaning,number.增加和減少refcount的規(guī)則和上面提到的一樣
PHP學(xué)習(xí)
特例,添加數(shù)組自己作為數(shù)組元素時:
PHP學(xué)習(xí)
<?php
$a = array('one');
$a[] = &$a;
xdebug_debug_zval('a');
?>
PHP學(xué)習(xí)
以上代碼輸出的結(jié)果:
PHP學(xué)習(xí)
a: (refcount=2, is_ref=1)=array (0 => (refcount=1, is_ref=0)='one', 1 => (refcount=2, is_ref=1)=...)
PHP學(xué)習(xí)
可以看到數(shù)組a和數(shù)組自己元素a[1]指向的變量容器refcount為2
PHP學(xué)習(xí)當(dāng)對數(shù)組$a調(diào)用unset函數(shù)時,$a的refcount變?yōu)?,發(fā)生了內(nèi)存泄漏
PHP學(xué)習(xí)清理變量容器的問題
盡管不再有某個作用域中的任何符號指向這個結(jié)構(gòu)(就是變量容器),由于數(shù)組元素"1"仍然指向數(shù)組本身,所以這個容器不能被消除.因為沒有另外的符號指向它,用戶沒有辦法清除這個結(jié)構(gòu),結(jié)果就會導(dǎo)致內(nèi)存泄漏.慶幸的是,php將在哀求結(jié)束時清除這個數(shù)據(jù)結(jié)構(gòu),但是php清除前,將耗費不少內(nèi)存空間
PHP學(xué)習(xí)
回收周期
5.3.0PHP使用了新的同步周期回收算法,來處理上面所說的內(nèi)存泄漏問題
PHP學(xué)習(xí)首先,我們先要建立一些基本規(guī)則:
如果一個引用計數(shù)增加,它將繼續(xù)被使用,當(dāng)然就不再垃圾中.如果引用技術(shù)減少到零,所在的變量容器將被清除(free).便是說,僅僅在引用計數(shù)減少到非零值時,才會產(chǎn)生垃圾周期(grabage cycle).其次,在一個垃圾周期中,通過檢查引用計數(shù)是否減1,并且檢查哪些變量容器的引用次數(shù)是零,來發(fā)現(xiàn)哪部分是垃圾
PHP學(xué)習(xí)
PHP學(xué)習(xí)為避免不得不檢查所有引用計數(shù)可能減少的垃圾周期,這個算法把所有可能根(possible roots 都是zval變量容器),放在根緩沖區(qū)(root buffer)中(用紫色標(biāo)志),這樣可以同時確保每個可能的垃圾根(possible garbage root)在緩沖區(qū)只出現(xiàn)一次.僅僅在根緩沖區(qū)滿了時,才對緩沖區(qū)內(nèi)部所有不同的變量容器執(zhí)行垃圾回收操作.
歡迎參與《PHP實戰(zhàn):解讀PHP中的垃圾回收機(jī)制》討論,分享您的想法,維易PHP學(xué)院為您提供專業(yè)教程。
轉(zhuǎn)載請注明本頁網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/8911.html