深入分析PHP Opcache工作原理
PHP專案中,尤其是在高並發大流量的場景中,如何提升PHP的回應時間,是一項十分重要的工作。
而Opcache又是優化PHP效能不可缺少的元件,尤其是應用了PHP框架的專案中,作用更是明顯。
1. 概述
在理解OPCache 功能之前,我們必須先理解PHP-FPM Nginx 的工作機制,以及PHP腳本解釋執行的機制。
1.1 PHP-FPM Nginx 的工作機制
請求從Web瀏覽器到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腳本,如果
- 沒有空閒的Worker,回傳502錯誤Worker(php-cgi)程序執行PHP腳本,如果
- 逾時,回傳504錯誤處理結束,回傳結果
PHP-FPM Worker 程序傳回處理結果,並關閉連接,等待下一個請求
- PHP-FPM Master 進程透過Socket 回傳處理結果
- Nginx Handler順序將每一個回應buffer傳送給第一個filter → 第二個→ 以此類推→ 最終回應傳送給客戶端
1.2 PHP腳本解釋執行的機制了解了PHP Nginx 整體的處理流程後,我們接下來來看PHP腳本具體執行流程,
首先我們來看一個實例:
<?php if (!empty($_POST)) { echo "Response Body POST: ", json_encode($_POST), "\n"; } if (!empty($_GET)) { echo "Response Body GET: ", json_encode($_GET), "\n"; }
我們來分析執行過程:
-
php初始化執行環節,#啟動Zend引擎,載入註冊的擴充模組
初始化後讀取腳本文件,Zend引擎對腳本文件進行詞法分析(lex),語法分析(bison),產生語法樹
Zend 引擎編譯語法樹,產生opcode#,
Zend 引擎執行opcode,傳回執行結果
在PHP cli模式下,每次執行PHP腳本,四個步驟都會依序執行一遍;
在PHP-FPM模式下,步驟1)在PHP-FPM啟動時執行一次,後續的請求中不再執行;步驟2)~4)每個請求都要執行一遍;
其實步驟2)、3)產生的語法樹和opcode,同一個PHP腳本每次執行的結果都是一樣的,
在PHP-FPM模式下,每次請求都要處理一遍,是對系統資源極大的浪費,那麼有沒有辦法優化呢?
當然有,如:
- OPCache:前身是Zend Optimizer ,是Zend Server 的開源元件;官方出品,強力推薦
- APC: Alternative PHP Cache 是一個開放自由的PHP opcode 快取元件,用於快取、最佳化PHP 中間程式碼;已經不更新了不推薦
- APCu:是APC的一個分支,共享內存,快取用戶數據,不能快取opcode ,可以配合Opcache 使用
- eAccelerate:同樣是不更新了,不推薦
- xCache:不再推薦使用了
2. OPCache 介紹
OPCache 是Zend官方出品的,開放自由的opcode 快取擴展,還具有程式碼最佳化功能,省去了每次載入和解析PHP 腳本的開銷。
PHP 5.5.0 及後續版本中已經綁定了 OPcache 擴充。
快取兩類內容:
- OPCode
- Interned String,如註解、變數名稱等
3. OPCache 原理
OPCache快取的機制主要是:將編譯好的操作碼放入共享內存,提供給其他進程存取#。
這裡就牽涉到記憶體共享機制,另外所有記憶體資源操作都有鎖定的問題,我們一一解讀。
3.1 共享記憶體
UNIX/Linux 系統提供許多進程間記憶體共享的方式:
- System-V shm API: System V共享記憶體,
- sysv shm是持久化的,除非被一個行程明確的刪除,否則它總是存在於記憶體裡,直到系統關機;
- mmap API:
- mmap映射的內存在不是持久化的,如果進程關閉,映射隨即失效,除非事先已經映射到了一個文件上
- 內存映射機制mmap是POSIX標準的系統調用,有匿名映射和檔案映射兩種
- mmap的一大優點是把檔案映射到進程的位址空間
- 避免了資料從用戶緩衝區到核心page cache緩衝區的複製過程;
- 當然還有一個優點就是不需要頻繁的read/write系統呼叫
##POSIX API: System V 的共享記憶體是過時的, POSIX共享內存提供了使用更簡單、設計更合理的API. - Unix socket API
當Wasted記憶體大於設定值時,自動重新啟動OPCache機制,清空並重新產生快取。
3.2 互斥鎖定
#任何記憶體資源的操作,都牽涉到鎖的機制。 共享記憶體:一個單位時間內,只允許一個程序執行寫入操作,允許多個程序執行讀取操作;寫入操作同時,不阻止讀取操作,以至於很少有鎖死的情況。 這就引發另一個問題:新程式碼、大流量場景,行程排隊執行快取opcode操作;重複寫入,導致資源浪費。4. OPCache 快取解讀
OPCache 是官方的Opcode 快取解決方案,在PHP5.5版本之後,已經打包到PHP原始碼中一起發布。 它將PHP編譯產生的字節碼以及資料快取到共享記憶體中, 在每次請求,從快取中直接讀取編譯後的opcode,進行執行。 透過節省腳本的編譯過程,提升PHP的運作效率。 如果正在使用APC擴展,做同樣的工作,現在強烈建議OPCache來代替,尤其是PHP7。4.1 OPCode 快取
#Opcache 會快取OPCode以及如下內容:- PHP腳本涉及的函數
- PHP腳本中定義的Class
- PHP腳本檔案路徑
- PHP腳本OPArray ##PHP腳本本身結構/內容
4.2 Interned String 快取
首先我們需要理解,什麼是Interned String? 在PHP5.4的時候, 引入了Interned String機制, 用於優化PHP對字串的儲存和處理。 尤其是處理大塊的字串,例如PHP doces時,Interned String 可以優化記憶體。 Interned String 快取的內容包括:變數名稱、類別名稱、方法名稱、字串、註解等。
在PHP-FPM模式中,Interned String 快取字符,僅限於Worker 進程內部。 而快取到OPCache中,那麼Worker進程之間可以使用 Interned String 快取的字串,節省記憶體。 我們要注意一個事情,在PHP開發中,一般會有大段的註釋,也會被快取到OPCache。
可以透過php.ini的配置,關閉註解的快取。 但是,像Zend Framework等框架中,會引用註釋,所以,是否關閉註釋的緩存,需要區別對待。
5. OPCache 更新策略
是緩存,都存在過期,以及更新策略等。 而OPCache的更新策略非常簡單,到期資料置為Wasted,達到設定值,清空緩存,重建快取。 這裡要注意:在高流量的場景下,重建快取是一件非常耗費資源的事兒。
OPCache 在建立快取時並不會阻止其他進程讀取。 這會導致大量進程反覆新建快取。所以,不要設定OPCache過期時間
每次發布新程式碼時,都會出現重複新建快取的情況。如何避免呢?- 不要在高峰期發布程式碼,這是任何情況下都要遵守的規則 程式碼預熱,例如使用腳本批次調PHP 存取URL,或使用OPCache 暴露的API 如
- opcache_compile_file()
進行編譯快取
#6.1 記憶體配置
- #opcache.preferred_memory_model="mmap"
- OPcache 首選的記憶體模組。如果留空,OPcache 會選擇適用的模組, 通常情況下,自動選擇就可以滿足需求。可選值包括:
mmap
,shm
,posix
以及win32
。 opcache.memory_consumption=64 - OPcache 的共享記憶體大小,以兆位元組為單位,預設
64M
#opcache.interned_strings_buffer =4 - 用來儲存臨時字串的記憶體大小,以兆位元組為單位,預設
4M
opcache.max_wasted_percentage=5 - 浪費記憶體的上限,以百分比計。如果達到此上限,那麼 OPcache 將產生重新啟動續發事件。預設
5
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。預設值
2000opcache.max_file_size=0
以位元組為單位的快取的檔案大小上限。設定為 0 表示快取全部檔案。預設值0
opcache.load_comments
- boolean
如果停用,則即使文件中包含註釋,也不會載入這些註釋內容。本選項可以和
opcache.save_comments一起使用,以實現按需載入註解內容。
opcache.fast_shutdown
boolean 如果啟用,則會使用快速停止續發事件。所謂快速停止續發事件是指依賴 Zend 引擎的記憶體管理模組 一次釋放全部請求變數的內存,而不是依序釋放每一個已分配的記憶體區塊。
# 推薦學習:《PHP影片教學》
opcache.file_cache
設定二級快取目錄並啟用二級快取。啟用二級快取可以在 SHM 記憶體滿了、伺服器重新啟動或重置 SHM 的時候提高效能。預設值為空字串 ""
,表示停用基於檔案的快取。 opcache.file_cache_only
boolean
啟用或停用在共享記憶體中的 opcode 快取。 opcache.file_cache_consistency_checks
boolean
當從檔案快取載入腳本的時候,是否對檔案的校驗和進行驗證。 opcache.file_cache_fallback
boolean
在Windows 平台上,當一個行程無法附加到共享記憶體的時候, 使用基於檔案的緩存,也即: opcache.file_cache_only=1
。需要顯示的啟用檔案快取。
以上是深入分析PHP Opcache工作原理的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。
