這篇文章為大家帶來了關於PHP opcache的相關知識,其中主要給大家聊一聊如何理解OPCache 功能,並且如何使用它,感興趣的朋友下面一起來看一下吧,希望對大家有幫助。
PHP專案中,尤其是在高並發大流量的場景中,如何提升PHP的回應時間,是一項十分重要的工作。
而Opcache又是優化PHP效能不可缺少的元件,尤其是應用了PHP框架的專案中,作用更是明顯。
1、概述
在理解 OPCache 功能之前,我們必須先理解PHP-FPM Nginx 的工作機制,以及PHP腳本解釋執行的機制。
1.1 PHP-FPM Nginx 的工作機制
請求從網頁瀏覽器到Nginx,再到PHP處理完成,總共要經歷如下五個步驟:
第一步驟:啟動服務
啟動PHP-FPM。 PHP-FPM 支援兩種通訊模式:TCP socket和Unix socket;
PHP-FPM 會啟動兩種類型的進程:Master 進程和Worker 進程,前者負責監控連接埠、分配任務、管理Worker進程;後者就是PHP的cgi程序,負責解釋編譯執行PHP腳本。
啟動Nginx。首先會載入ngx_http_fastcgi_module 模組,初始化FastCGI執行環境,實作FastCGI協定請求代理
這裡要注意:fastcgi的worker進程(cgi進程),是由PHP-FPM來管理,不是Nginx。 Nginx只是代理
第二步:Request => Nginx
Nginx 接收請求,並基於location配置,選擇一個合適handler
這裡就是代理PHP的handler
#第三步:Nginx => PHP-FPM
Nginx 把請求翻譯成fastcgi請求
透過TCP socket/Unix Socket 傳送給PHP-FPM 的master進程
第四步:PHP-FPM Master => Worker
PHP-FPM master 程式接收到請求
處理結束,回傳結果
第五步:PHP-FPM Worker => Master => ; Nginx
##PHP-FPM Master進程透過Socket 回傳處理結果
Nginx Handler順序將每一個回應buffer傳送給第一個filter → 第二個→ 以此類推→ 最終回應傳送給客戶端
1.2 PHP腳本解釋執行的機制了解了PHP Nginx 整體的處理流程後,我們接下來看一下PHP腳本具體執行流程,首先我們來看一個實例:
<p><?php <br> if (!empty($_POST)) {<br> echo "Response Body POST: ", json_encode($_POST), "\n";<br> }<br> if (!empty($_GET)) { <br> echo "Response Body GET: ", json_encode($_GET), "\n";<br> }<br></p>
2.初始化後讀取腳本文件,Zend引擎對腳本檔案進行詞法分析(lex),語法分析(bison),產生語法樹
3.Zend 引擎編譯語法樹,產生opcode,
4.Zend 引擎執行opcode,傳回執行結果
在PHP cli模式下,每次執行PHP腳本,四個步驟都會依序執行一遍;在PHP-FPM模式下,步驟1)在PHP-FPM啟動時執行一次,後續的請求中不再執行;步驟2)~4)每個請求都要執行一遍;
OPCache:前身是Zend Optimizer ,是Zend Server 的開源元件;官方出品,強力推薦
APC:Alternative PHP Cache 是開放自由的PHP opcode 快取元件,用於快取、最佳化PHP 中間程式碼;已經不更新了不推薦
APCu:是APC的一個分支,共享內存,快取用戶數據,不能緩存opcode,可以配合Opcache 使用
eAccelerate:同樣是不更新了,不推薦
xCache:不再推薦使用了
2. OPCache 介紹
快取兩類內容:
OPCode
#Interned String,如註解、變數名稱等
3 . OPCache 原理###OPCache快取的機制主要是:將編譯好的操作碼放入共享內存,提供給其他進程存取。這裡就牽涉到記憶體共享機制,另外所有記憶體資源操作都有鎖的問題,我們一一解讀。 ######3.1 共享記憶體###UNIX/Linux 系統提供許多進程間記憶體共享的方式:
1.System-V shm API: System V共享記憶體
sysv shm是持久化的,除非被一個進程明確的刪除,否則它始終存在於記憶體裡,直到系統關機;
2.mmap API:
#mmap映射的內存在不是持久化的,如果進程關閉,映射隨即失效,除非事先已經映射到了一個檔案上
記憶體映射機制mmap是POSIX標準的系統調用,有匿名映射和檔案映射兩種
System V 的共享記憶體是過時的, POSIX共享記憶體提供了使用更簡單、設計更合理的API.
當Wasted記憶體大於設定值時,自動重新啟動OPCache機制,清空並重新產生快取。
共享記憶體:一個單位時間內,只允許一個程序執行寫入操作,允許多個程序執行讀取操作;寫入操作同時,不阻止讀取操作,以至於很少有鎖死的情況。
這就引發另一個問題:新程式碼、大流量場景,行程排隊執行快取opcode操作;重複寫入,導致資源浪費。
4. OPCache 快取解讀
OPCache 是官方的Opcode 快取解決方案,在PHP5.5版本之後,已經打包到PHP原始碼中一起發布。它將PHP編譯產生的字節碼以及資料快取到共享記憶體中, 在每次請求,從快取中直接讀取編譯後的opcode,進行執行。
透過節省腳本的編譯過程,提高PHP的運作效率。如果正在使用APC擴展,做同樣的工作,現在強烈建議OPCache來代替,尤其是PHP7。
快取首先我們需要理解,什麼是Interned String? 在PHP5.4的時候, 引入了Interned String機制, 用於優化PHP對字串的儲存和處理。尤其是處理大塊的字串,例如PHP doces時,Interned String 可以優化記憶體。 Interned String 快取的內容包括:變數名稱、類別名稱、方法名稱、字串、註解等。
但是,像Zend Framework等框架中,會引用註釋,所以,是否關閉註釋的緩存,需要區別對待。
5. OPCache 更新策略
是緩存,都存在過期,以及更新策略等。而OPCache的更新策略非常簡單,到期資料置為Wasted,達到設定值,清空緩存,重建快取。這裡要注意:在高流量的場景下,重建快取是一件非常耗費資源的事兒。 OPCache 在創建快取時並不會阻止其他進程讀取。這會導致大量進程反覆新建快取。所以,不要設定OPCache過期時間
#6.1 記憶體配置
6.2 允許快取的檔案數量以及大小
opcache.max_accelerated_files=2000 OPcache 哈希表中可儲存的腳本檔案數量上限。真實的取值是在質數集合 { 223, 463, 983, 1979, 3907, 7963, 16229, 32531, 65407, 130987 } 中找到的第一個大於等於設定值的質數。設定值取值範圍最小值是 200,最大值在 PHP 5.5.6 之前是 100000,PHP 5.5.6 之後是 1000000。預設值2000
opcache.max_file_size=0 以位元組為單位的快取的檔案大小上限。設定為 0 表示快取全部檔案。預設值0
6.3 註解相關的快取
opcache.load_commentsboolean 如果停用,則即使檔案中包含註釋,也不會載入這些註釋內容。本選項可以和 opcache.save_comments 一起使用,以實現按需載入註解內容。
opcache.fast_shutdown boolean 如果啟用,則會使用快速停止續發事件。所謂快速停止續發事件是指依賴 Zend 引擎的記憶體管理模組 一次釋放全部請求變數的內存,而不是依序釋放每一個已分配的記憶體區塊。
6.4 二級快取的設定
#opcache.file_cache 設定二級快取目錄並啟用二級快取。啟用二級快取可以在 SHM 記憶體滿了、伺服器重新啟動或重置 SHM 的時候提高效能。預設值為空字串 "",表示禁用基於檔案的快取。
opcache.file_cache_onlyboolean 啟用或停用在共享記憶體中的 opcode 快取。
opcache.file_cache_consistency_checksboolean 當從檔案快取載入腳本的時候,是否對檔案的校驗和進行驗證。
opcache.file_cache_fallbackboolean 在 Windows 平台上,當一個程序無法附加到共享記憶體的時候, 使用基於檔案的緩存,也即:opcache.file_cache_only=1。需要顯示的啟用檔案快取。
推薦學習:《PHP影片教學》
以上是一文詳解PHP opcache的原理及使用的詳細內容。更多資訊請關注PHP中文網其他相關文章!