PHP 有一個非常簡單的垃圾收集器,它實際上將對不再位於內存范圍(scope)中的對象進行垃圾收集。垃圾收集的內部方式是使用一個引用計數器,因此當計數器達到 0 時(意味著對該對象的引用都不可用),對象將被當作垃圾收集並從內存中刪除。
每一種計算機語言都有自己的自動垃圾回收機制,讓程序員不必過分關心程序內存分配,php也不例外,但是在面向對象編程(OOP)編程中,有些對象需要顯式的銷毀;防止程序執行內存溢出。
一、PHP 垃圾回收機制(Garbage Collector 簡稱GC)
在PHP中,沒有任何變量指向這個對象時,這個對象就成為垃圾。PHP會將其在內存中銷毀;這是PHP的GC垃圾處理機制,防止內存溢出。
當一個PHP線程結束時,當前占用的所有內存空間都會被銷毀,當前程序中所有對象同時被銷毀。GC進程一般都跟著每起一個SESSION而開始運行的.gc目的是為了在session文件過期以後自動銷毀刪除這些文件.
二、__destruct /unset
__destruct() 析構函數,是在垃圾對象被回收時執行。
unset 銷毀的是指向對象的變量,而不是這個對象。
三、 Session 與 GC
由於PHP的工作機制,它並沒有一個daemon線程來定期的掃描Session信息並判斷其是否失效,當一個有效的請求發生時,PHP 會根據全局變量 session.gc_probability和session.gc_divisor的值,來決定是否啟用一個GC, 在默認情況下,session.gc_probability=1, session.gc_divisor =100也就是說有1%的可能性啟動GC(也就是說100個請求中只有一個gc會伴隨100個中的某個請求而啟動).
GC的工作就是掃描所有的Session信息,用當前時間減去session最後修改的時間,同session.gc_maxlifetime參數進行比較,如果生存時間超過gc_maxlifetime(默認24分鐘),就將該session刪除。
但是,如果你Web服務器有多個站點,多個站點時,GC處理session可能會出現意想不到的結果,原因就是:GC在工作時,並不會區分不同站點的session.
那麼這個時候怎麼解決呢?
1. 修改session.save_path,或使用session_save_path()讓每個站點的session保存到一個專用目錄,
2. 提供GC的啟動率,自然,GC的啟動率提高,系統的性能也會相應減低,不推薦。
3. 在代碼中判斷當前session的生存時間,利用session_destroy()刪除