解決PHP Opcache 緩存刷新、代碼重載出現(xiàn)無法更新代碼的問題
問題背景
通過啟用Opcache的緩存優(yōu)化,將PHP代碼預(yù)編譯為Opcode緩存到共享內(nèi)存中供進程反復(fù)調(diào)用,從而減少了重復(fù)從磁盤解析PHP代碼的時間消耗,顯著的提高了PHP性能,提升了業(yè)務(wù)性能的調(diào)用,但是也引發(fā)了一些問題,就是我們每次更新了相應(yīng)的PHP代碼后,web server 無法即時加載到更新后的代碼。
解決方案
(一)、設(shè)置Opcache腳本驗證時間
可以通過更改 Opcache 以下兩個配置選項來調(diào)整代碼重載時間
opcache.revalidate_freq=0 檢查腳本時間戳是否有更新的周期,以秒為單位。(如果設(shè)置為 0 會導致針對每個請求, OPcache 都會檢查腳本更新)
opcache.validate_timestamps=0 如果啟用,那么 OPcache 會每隔 opcache.revalidate_freq 設(shè)定的秒數(shù) 檢查腳本是否更新。
PS:在實際生產(chǎn)環(huán)境中,為了盡可能達到最優(yōu)性能,盡量不開啟文件更新驗證,因為每次驗證都會重新預(yù)編譯PHP代碼到共享內(nèi)存中。
(二)、重啟 | 重載 php-fpm 進程
每次重啟或重啟 php-fpm 進程便會重新解析PHP腳本文件,但是重啟 fpm 進程可能會導致請求中斷,從而導致寫入臟數(shù)據(jù) 或者 造成事務(wù)回滾等一系列異常。
重載相對于重啟則平順很多,不會導致用戶請求直接中斷,相對來說風險低很多,但是php-fpm 收到reload信號,便會向所有子進程發(fā)送SIGGUIT信號,同時注冊一個定時器,在規(guī)定的時間之內(nèi)子進程沒有退出,接著在發(fā)送SIGTERM信號,結(jié)束子進程。如果在一秒之內(nèi)子進程還是沒結(jié)束 直接發(fā)送SIGKILL 強制殺死。
重啟php-fpm
service php-fpm restart
重載php-fpm
services php-fpm reload 或 kill -USR2 `cat /usr/local/php/var/run/php-fpm.pid`
(三)、手動清理緩存
除了上面的兩種方式,還有更為穩(wěn)妥一點的緩存清理,我們可以通過opcache_reset()和opcache_invalidate() 函數(shù)來刷新Opcache緩存。
opcache_reset()
- 重置整個Opcode緩存,所有的PHP腳本將會被重新解析再預(yù)編譯為Opcode。
opcache_invalidate()
- 清除指定腳本緩存,可以傳遞兩個參數(shù),一個是刷新文件路徑,一個是force字段, 如果 force 沒有設(shè)置或者傳入的是 FALSE,那么只有當腳本的修改時間 比對應(yīng)Opcode的時間更新時,腳本的緩存才會失效。
需要注意的是,當PHP以PHP-FPM的方式運行的時候,opcache的緩存是無法通過php命令進行清除的,只能通過http或cgi到php-fpm進程的方式來清除緩存,我們可以編寫一個對外接口,來達到清理緩存的目的。
相關(guān)實現(xiàn)如下(框架:laravel):
Route::any('cache-reset', function () { //重置整個Opcode緩存 dd(opcache_reset()); }); Route::any('cache-update', function () { //清除掉最近一次更新文件的緩存 exec('git diff --name-only HEAD~ HEAD', $output); foreach ($output as $file) { $path = base_path($file); opcache_invalidate($path, true); } dd('刷新完成'); });
總結(jié)
通過上面的三種策略,可以實現(xiàn) Opcache 緩存更新的目的,但是在流量高峰期或者大流量的服務(wù)端,每次更新緩存都是一件非常損耗資源的事情,Opcache在重建緩存時,也不會禁止其他進程讀取,因此就會造成反復(fù)新建緩存,因此想要達到最佳的性能調(diào)配:
- 最好不要在高峰期清理緩存
- 高峰期不要頻繁的更新代碼,清理緩存,會造成重復(fù)新建緩存
- 如果需要更新,可以嘗試削弱服務(wù)端權(quán)重,實現(xiàn)逐個更新的目的。
- 如果需要強制更新,盡量選擇手動清除緩存的方式,來重建Opcache緩存,使代價最小化。
以上就是解決PHP Opcache 緩存刷新、代碼重載出現(xiàn)無法更新代碼的問題的詳細內(nèi)容,更多關(guān)于PHP Opcache 緩存刷新、代碼重載的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
PHP中實現(xiàn)漢字轉(zhuǎn)區(qū)位碼應(yīng)用源碼實例解析
PHP里如何實現(xiàn)漢字轉(zhuǎn)區(qū)位碼這個問題一直困擾這大多程序員,那么下面這個源碼實例相信能給大家?guī)砗艽蟮膸椭?/div> 2010-06-06php heredoc和phpwind的模板技術(shù)使用方法小結(jié)
Heredoc技術(shù),在正規(guī)的PHP文檔中和技術(shù)書籍中一般沒有詳細講述,只是提到了這是一種Perl風格的字符串輸出技術(shù)。但是現(xiàn)在的一些論壇程序,和部分文章系統(tǒng),都巧妙的使用heredoc技術(shù),來部分的實現(xiàn)了界面與代碼的準分離,phpwind就是一個典型的例子。2008-03-03PHP封裝的svn類使用內(nèi)置svn函數(shù)實現(xiàn)根據(jù)svn版本號導出相關(guān)文件示例
這篇文章主要介紹了PHP封裝的svn類使用內(nèi)置svn函數(shù)實現(xiàn)根據(jù)svn版本號導出相關(guān)文件,結(jié)合實例形式分析了php封裝的svn操作類與根據(jù)版本導出相關(guān)版本文件操作技巧,需要的朋友可以參考下2018-06-06php實現(xiàn)刪除指定目錄下相關(guān)文件的方法
這篇文章主要介紹了php實現(xiàn)刪除指定目錄下相關(guān)文件的方法,主要涉及對文件的遍歷以及對文件的各種常用操作,需要的朋友可以參考下2014-10-10php實現(xiàn)的Curl封裝類Curl.class.php用法實例分析
這篇文章主要介紹了php實現(xiàn)的Curl封裝類Curl.class.php用法,以完整實例形式較為詳細的分析了Curl封裝類的定義及相關(guān)使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-09-09最新評論