您好,登錄后才能下訂單哦!
這篇文章主要介紹PHP中垃圾回收相關函數的使用案例,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
php的框架:1、Laravel,Laravel是一款免費并且開源的PHP應用框架。2、Phalcon,Phalcon是運行速度最快的一個PHP框架。3、Symfony,Symfony是一款為Web項目準備的PHP框架。4、Yii,Yii是一款快速、安全和專業的PHP框架。5、CodeIgniter,CodeIgniter是一款非常敏捷的開源PHP框架。6、CakePHP,CakePHP是一款老牌的PHP框架。7.Kohana,Kohana是一款敏捷但是功能強大的PHP框架。
我們為什么要強調 “循環引用” 呢?其實,在默認情況下,我們直接 unset() 掉一個沒有被其他變量引用的變量時,就會讓這個變量的引用計數變為0。這時,PHP 默認的垃圾回收機制就會直接清除掉這個變量。比如:
$a = new stdClass; $b = new stdClass; $c = new stdClass; echo memory_get_usage(), PHP_EOL; // 706528 unset($a); echo memory_get_usage(), PHP_EOL; // 706488 gc_collect_cycles(); echo memory_get_usage(), PHP_EOL; // 706488
從上面的代碼中可以看出,我們 unset() 掉 $a 之后,內存直接就減少了。但是,如果是產生了循環引用的情況,那么簡單的進行 unset() 就沒有效果了。
class D{ public $d; } $d = new D; $d->d = $d; echo memory_get_usage(), PHP_EOL; // 706544 unset($d); echo memory_get_usage(), PHP_EOL; // 706544 gc_collect_cycles(); echo memory_get_usage(), PHP_EOL; // 706488
在這段代碼中,我們對 \$d 進行了一個簡單的循環引用賦值。使用 unset() 后,內存沒有發生變化,這時,只能使用 gc_collect_cycles() 函數來進行強制的循環引用清理,才能將 $d 里面的無效循環引用清除掉。
沒錯,這一段的重點正是 gc_collect_cycles() 這個函數。它在正常情況下對普通的變量引用是不會產生什么清理效果的,當然,對于普通的變量我們直接 unset() 掉就可以了。它最主要的作用就是針對循環引用的清理。之前我們學習過,循環引用計數會存在一個 根緩沖區 ,一般默認情況下它能容納 10000 個待清理的 可能根 。而 gc_collect_cycles() 的作用就是不用等這個 根緩沖區 滿就直接進行清理(個人理解)。關于這個垃圾回收算法的內容請移步:PHP垃圾回收機制的一些淺薄理解
其實,大部分情況下我們是不太需要關注 PHP 的垃圾回收問題的,也就是說,我們不是很需要手動地去調用這個 gc_collect_cycles() 函數。PHP-FPM 在每次調用完成后會直接整體的釋放,簡單的一次 CLI 腳本執行完也會全部釋放。沒錯,正常情況下,PHP 一次執行完成之后就會銷毀所有的內容,內存垃圾自然也就不存在了。但是,在執行長時間的守護腳本時,或者使用常駐進程的框架(Swoole)時,還是需要注意有沒有循環引用的問題。因為這種程序一直運行,如果存在大量循環引用對象時,就有可能導致內存泄露。
gc_disable(); echo gc_enabled(), PHP_EOL; // gc_enable(); echo gc_enabled(), PHP_EOL; // 1
很簡單的三個函數,gc_disable() 是 “停用循環引用收集器”,gc_enable() 是“開啟循環引用收集器”,而 gc_enabled() 就是查看當前的循環引用收集器是否開啟。
gc_mem_caches()
官網及網絡上并沒有什么詳細的介紹,不過從定義來看,它主要的作用就是回收 PHP 底層的 Zend 引擎內存管理器所使用過的內存。這個大家了解下就好,平常也從來沒用過。
$e = new stdClass; for($i = 100;$i>0;$i--){ $e->list[] = $e; } unset($e); gc_collect_cycles(); var_dump(gc_status()); // array(4) { // ["runs"]=>int(1) // ["collected"]=>int(2) // ["threshold"]=>int(10001) // ["roots"]=>int(0) // }
我們還是做了一個循環引用的對象,然后使用 gc_status() 來查看當前垃圾回收器中關于循環引用的狀態。從返回的內容可以看出, runs 運行了 1 個,collected 收集了 2 個, threshold 閾值是 10001,roots 可能根沒有了(已經被回收了)。
這個函數可以在測試環境中對代碼的運行情況進行檢查,查看我們代碼中有沒有不正常的循環引用情況,當然,上面的解釋也只是個人的推測,因為關于這方面的資料確實非常少。
以上是“PHP中垃圾回收相關函數的使用案例”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。