摘要:后者的例子在給巨大的一個知名的組件庫的模板組件做單元測試時,就可能會出現(xiàn)問題。
PHP的垃圾回收機制
垃圾回收器,全稱Garbage Collection,簡稱GC,5.3版本之前 只是簡單判斷變量的zval的refcount是否為0,是的話就釋放,不是直至進程結(jié)束(隱藏著變量內(nèi)存溢出的風險).
如果你已經(jīng)安裝了? Xdebug,你能通過調(diào)用函數(shù) xdebug_debug_zval()顯示"refcount"和"is_ref"的值。
引用計數(shù)基本知識 1、一個zval變量容器:內(nèi)容 | ||
---|---|---|
類型 | ||
值 | ||
is_ref | 是個bool值,用來標識這個變量是否是屬于引用集合(reference set) | zval變量容器中還有一個內(nèi)部引用計數(shù)機制,來優(yōu)化內(nèi)存使用 |
refcount | 指向這個zval變量容器的變量(也稱符號即symbol)個數(shù) | 所有的符號存在一個符號表中,其中每個符號都有作用域(scope),那些主腳本(比如:通過瀏覽器請求的的腳本)和每個函數(shù)或者方法也都有作用域。 |
當一個變量被賦常量值時,就會生成一個zval變量容器
1、 生成一個新的zval容器
參考上面的定義,分別是:
生成的變量容器,其中:
類型 string
值 new string
is_ref false
Refcount 1
2、增加一個zval的引用計數(shù)把一個變量賦值給另一變量將增加引用次數(shù)(refcount).
類型 string
值 new string
is_ref true
Refcount 2
引用次數(shù)是2,因為同一個變量容器被變量 a 和變量 b關(guān)聯(lián).當沒必要時,php不會去復制已生成的變量容器。變量容器在”refcount“變成0時就被銷毀. 當任何關(guān)聯(lián)到某個變量容器的變量離開它的作用域(比如:函數(shù)執(zhí)行結(jié)束),或者對變量調(diào)用了函數(shù) unset()時,”refcount“就會減1.
3、 減少引用計數(shù)
a: (refcount=3, is_ref=0)="new string"
a: (refcount=1, is_ref=0)="new string"
如果我們現(xiàn)在執(zhí)行 unset($a);,包含類型和值的這個變量容器就會從內(nèi)存中刪除。
4、復合類型(Compound Types) ?當考慮像 array和object這樣的復合類型時,事情就稍微有點復雜. 與 標量(scalar)類型的值不同,array和 object類型的變量把它們的成員或?qū)傩源嬖谧约旱姆柋碇?。這意味著下面的例子將生成三個zval變量容器。
"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
)
看圖說話
這三個zval變量容器是: a,meaning和 number。增加和減少”refcount”的規(guī)則和上面提到的一樣. 下面, 我們在數(shù)組中再添加一個元素,并且把它的值設(shè)為數(shù)組中已存在元素的值:
添加一個已經(jīng)存在的元素到數(shù)組中
"life", "number" => 42 );
$a["life"] = $a["meaning"];
xdebug_debug_zval( "a" );
?>
輸出
a: (refcount=1, is_ref=0)=array (
"meaning" => (refcount=2, is_ref=0)="life",
"number" => (refcount=1, is_ref=0)=42,
"life" => (refcount=2, is_ref=0)="life"
)
有圖有真相
從以上的xdebug輸出信息,我們看到原有的數(shù)組元素和新添加的數(shù)組元素關(guān)聯(lián)到同一個"refcount"2的zval變量容器. 盡管 Xdebug的輸出顯示兩個值為"life"的 zval 變量容器,其實是同一個。 函數(shù)xdebug_debug_zval()不顯示這個信息,但是你能通過顯示內(nèi)存指針信息來看到。
刪除數(shù)組中的一個元素,就是類似于從作用域中刪除一個變量. 刪除后,數(shù)組中的這個元素所在的容器的“refcount”值減少,同樣,當“refcount”為0時,這個變量容器就從內(nèi)存中被刪除,下面又一個例子可以說明:
"life", "number" => 42 );
$a["life"] = $a["meaning"];
unset( $a["meaning"], $a["number"] );
xdebug_debug_zval( "a" );
?>
輸出
a: (refcount=1, is_ref=0)=array (
"life" => (refcount=1, is_ref=0)="life"
)
現(xiàn)在,當我們添加一個數(shù)組本身作為這個數(shù)組的元素時,事情就變得有趣,下個例子將說明這個。例中我們加入了引用操作符,否則php將生成一個復制。
把數(shù)組作為一個元素添加到自己
輸出
a: (refcount=2, is_ref=1)=array (
0 => (refcount=1, is_ref=0)="one",
1 => (refcount=2, is_ref=1)=...
)
看圖說話
能看到數(shù)組變量 (a) 同時也是這個數(shù)組的第二個元素(1) 指向的變量容器中“refcount”為 2。上面的輸出結(jié)果中的"..."說明發(fā)生了遞歸操作, 顯然在這種情況下意味著"..."指向原始數(shù)組。
跟剛剛一樣,對一個變量調(diào)用unset,將刪除這個符號,且它指向的變量容器中的引用次數(shù)也減1。所以,如果我們在執(zhí)行完上面的代碼后,對變量$a調(diào)用unset, 那么變量 $a 和數(shù)組元素 "1" 所指向的變量容器的引用次數(shù)減1, 從"2"變成"1". 下例可以說明:
輸出
(refcount=1, is_ref=1)=array (
0 => (refcount=1, is_ref=0)="one",
1 => (refcount=1, is_ref=1)=...
)
看圖說話
2、清理變量容器的問題(Cleanup Problems) ?
盡管不再有某個作用域中的任何符號指向這個結(jié)構(gòu)(就是變量容器),由于數(shù)組元素“1”仍然指向數(shù)組本身,所以這個容器不能被清除 。因為沒有另外的符號指向它,用戶沒有辦法清除這個結(jié)構(gòu),結(jié)果就會導致內(nèi)存泄漏。慶幸的是,php將在腳本執(zhí)行結(jié)束時清除這個數(shù)據(jù)結(jié)構(gòu),但是在php清除之前,將耗費不少內(nèi)存。如果你要實現(xiàn)分析算法,或者要做其他像一個子元素指向它的父元素這樣的事情,這種情況就會經(jīng)常發(fā)生。當然,同樣的情況也會發(fā)生在對象上,實際上對象更有可能出現(xiàn)這種情況,因為對象總是隱式的被引用。
如果上面的情況發(fā)生僅僅一兩次倒沒什么,但是如果出現(xiàn)幾千次,甚至幾十萬次的內(nèi)存泄漏,這顯然是個大問題。這樣的問題往往發(fā)生在長時間運行的腳本中,比如請求基本上不會結(jié)束的守護進程(deamons)或者單元測試中的大的套件(sets)中。后者的例子:在給巨大的eZ(一個知名的PHP Library) 組件庫的模板組件做單元測試時,就可能會出現(xiàn)問題。有時測試可能需要耗用2GB的內(nèi)存,而測試服務(wù)器很可能沒有這么大的內(nèi)存。
PHP手冊
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/31004.html
摘要:本書的地址篇收集了一些常見的基礎(chǔ)進階面試題,基礎(chǔ)的面試題不再作答。如何實現(xiàn)持久化持久化,將在內(nèi)存中的的狀態(tài)保存到硬盤中,相當于備份數(shù)據(jù)庫狀態(tài)。相當于備份數(shù)據(jù)庫接收到的命令,所有被寫入的命令都是以的協(xié)議格式來保存的。 本書的 GitHub 地址:https://github.com/todayqq/PH... PHP 篇收集了一些常見的基礎(chǔ)、進階面試題,基礎(chǔ)的面試題不再作答。 基礎(chǔ)篇 ...
摘要:所有這些類型,在內(nèi)部統(tǒng)一用一個叫做的結(jié)構(gòu)表示,在源代碼中這個結(jié)構(gòu)名稱為。的具體定義在源代碼的文件中,下面是相關(guān)代碼的摘錄。 【轉(zhuǎn)】淺談PHP5中垃圾回收算法(Garbage Collection)的演化 前言 PHP是一門托管型語言,在PHP編程中程序員不需要手工處理內(nèi)存資源的分配與釋放(使用C編寫PHP或Zend擴展除外),這就意味著PHP本身實現(xiàn)了垃圾回收機制(Garbage C...
摘要:內(nèi)部通過引用計數(shù)機制來統(tǒng)計一個對象被引用的次數(shù)。下一步,就該被我們的垃圾回收器給收走了。而我們垃圾回收機制只有當引用計數(shù)為的時候才會釋放對象。以空間換時間的方法提高垃圾回收效率。 人生苦短,只談風月,談什么垃圾回收。據(jù)說上圖是某語言的垃圾回收機制。。。我們寫過C語言、C++的朋友都知道,我們的C語言是沒有垃圾回...
閱讀 1671·2021-11-23 10:07
閱讀 2667·2019-08-30 11:10
閱讀 2851·2019-08-29 17:08
閱讀 1793·2019-08-29 15:42
閱讀 3189·2019-08-29 12:57
閱讀 2406·2019-08-28 18:06
閱讀 3555·2019-08-27 10:56
閱讀 395·2019-08-26 11:33