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