首頁 後端開發 php教程 了解一下PHP 8的 JIT 特性!

了解一下PHP 8的 JIT 特性!

Jul 21, 2020 pm 05:11 PM
jit

本篇文章要為大家介紹一下PHP 8 的 JIT特性。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有幫助。

TL;DR

PHP 8 的JIT(Just In Time)編譯器將作為擴展整合到php 中 Opcache 擴展用於運行時將某些操作碼直接轉換為從cpu 指令。

這表示使用 JIT 後,Zend VM 不需要解釋某些操作碼,而這些指令將直接作為 CPU 層級指令執行。

PHP 8 的 JIT

PHP 8 Just In Time (JIT) 編譯器帶來的影響是毋庸置疑的。但到目前為止,我發現關於 JIT 應該做什麼卻知之甚少。

經過多次研究和放棄,我決定親自檢查 PHP 原始碼。結合我對 C 語言的一些知識和我目前收集到的所有零散信息,我提出了這篇文章,我希望它能幫助您更好地理解 PHP 的 JIT。

簡單一點來說 : 當 JIT 按預期工作時,您的程式碼不會透過 Zend VM 執行,而是作為一組 CPU 層級指令直接執行。

這就是全部的想法。

但為了更好地理解它,我們需要考慮 php 如何在內部運作。不是很複雜,但需要一些介紹。

PHP 的程式碼是怎麼執行的?

總所周知, PHP 是解釋型語言,但這句話本身又是什麼意思呢?

每次執行 PHP 程式碼(命令列腳本或 WEB 應用程式)時,都要經過 PHP 解譯器。最常用的是 PHP-FPM 和 CLI 解譯器。

解釋器的工作很簡單:接收 PHP 程式碼,對其進行解釋,然後傳回結果。

一般的解釋型語言都是這個流程。有些語言可能會減少幾個步驟,但總體的思路相同。在 PHP 中,這個流程如下:

  1. 讀取 PHP 程式碼並將其解釋為一組稱為 Tokens 的關鍵字。這個過程讓解釋器知道各個程式都寫了哪些程式碼。 這一步稱為 Lexing 或 Tokenizing 。

  2. 拿到 Tokens 集合以後,PHP 解譯器會嘗試解析他們。透過稱為 Parsing 的過程產生抽象語法樹(AST)。這裡 AST 是一個節點集表示要執行哪些動作。例如,「 echo 1 1 」實際意義是 「印 1 1 的結果」 或者更詳細的說 「印出一個動作,這個動作是 1 1」。

  3. 有了 AST ,可以更輕鬆地理解操作和優先順序。將抽象語法樹轉換成可以被 CPU 執行的操作需要一個用於過渡的表達式 (IR),在 PHP 中我們稱為 Opcodes 。將 AST 轉換為 Opcodes 的過程稱為 compilation 。

  4. 有了 Opcodes ,有趣的部分就來了: executing 程式碼! PHP 有一個稱為 Zend VM 的引擎,該引擎能夠接​​收一系列 Opcodes 並執行它們。執行所有 Opcodes 後, Zend VM 就會將程式終止。

這個圖可以讓你更清楚:

#一個簡化版的 PHP 解釋流程概述。

如你所見。這裡有個問題:即使 PHP 程式碼沒改變,每次執行還是會走此流程嗎?

讓我們看回 Opcodes 。對了!這就是 Opcache 擴展 存在的原因。

Opcache 擴充功能

Opcache 擴充功能是 PHP 隨附的,通常沒必要停用它。使用 PHP 最好開啟 Opcache 。

它的作用是為 Opcodes 增加一個記憶體共享快取層。它的工作是從 AST 中提取新生成的 Opcodes 並緩存它們,以便執行時

可以跳過 Lexing/Tokenizing 和 Parsing 步驟。

這是包含 Opcache 擴充功能的流程示意圖:

PHP 使用 Opcache 的解釋流程。如果檔案已經被解析,則 PHP 會為其取得快取的 Opcodes ,而不是再次解析。

完美的跳過了 Lexing/Tokenizing 、 Parsing 和 Compiling 步驟 。

附註: 這是超讚的 PHP 7.4 預先載入功能 RFC ! 允許你告訴 PHP FPM 解析程式碼庫,將其轉換為 Opcodes 並且在執行之前就將其快取。

你想知道 JIT 是怎麼參與這個解釋流程的嗎?這篇文章的將說明。

Just In Time 編譯有什麼效果?

聽了 Zeev 在 PHP Internals News 發表的 PHP 和 JIT 廣播 之後,我弄清楚了 JIT 實際上做了什麼事情。

如果說 Opcache 擴充功能可以更快的取得 Opcodes 將其直接轉到 Zend VM,則 JIT 讓它們完全不使用 Zend VM 即可運作。

Zend VM 是用 C 寫的程序,可作為 Opcodes 和 CPU 之間的一層。 JIT 在執行時間直接產生編譯後的程式碼,因此 PHP 可以

跳過 Zend VM 並直接被 CPU 執行。 從理論上來說,性能會更好。

這聽起來很奇怪,因為在編譯成機器碼之前,需要為每種類型的結構體寫一個具體的實作。但其實這也是合理的。

PHP 的 JIT 使用了名為 DynASM (Dynamic Assembler) 的函式庫,該函式庫將特定格式的一組 CPU 指令對應為許多不同 CPU 類型的彙編程式碼。因此,編譯器只需要使用 DynASM 就可以將 Opcodes 轉換為特定結構體的機器碼。

但是,有一個問題困擾了我很久。

如果預先載入能夠在執行之前將PHP 程式碼解析為Opcodes,而DynASM 可以將Opcodes 編譯為機器碼(Just In Time 編譯) ,為什麼我們不立即使用執行前編譯(Ahead of Time 編譯) 立刻編譯PHP 呢?

透過收聽 Zeev 的廣播,我找到的原因之一是 PHP 是弱型別語言,這表示在 Zend VM 嘗試執行某個操作碼之前, PHP 通常不知道變數的型別。

可以查看 Zend_value 聯合類型 得知,許多指標指向不同類型的變數。每當 Zend VM 嘗試從 Zend_value 取得值時,它都會使用像 ZSTR_VAL 這樣的宏,取得聯合類型中字串的指標。

例如,這個 Zend VM handler 是處理「小於或等於」(<=) 表達式。看看它編碼這麼多的 if else 分支,只是為了類型推論。

使用機器碼執行類型推斷邏輯是不可行的,並且可能變得更慢。

先求值再編譯也不是個好選擇,因為編譯成機器碼就是 CPU 密集型任務。因此,在運行時編譯所有內容也不好。

那麼 Just In Time 編譯是怎麼做的?

現在我們知道無法很好的推斷類型來提前編譯。我們也知道在運行時進行編譯的運算成本很高。那麼 JIT 對 PHP 有何好處呢?

為了尋求平衡, PHP 的 JIT 嘗試只編譯有價值的 Opcodes 。為此, JIT 會分析 Zend VM 要執行的 Opcodes 並檢查可能編譯的地方。 (根據設定檔)

當某個 Opcode 編譯後,它將把執行交給該編譯後的程式碼,而不是交給 Zend VM 。看起來如下:

PHP 的 JIT 解釋流程。如果已編譯,則 Opcodes 不會透過 Zend VM 執行。

因此,在 Opcache 擴充中,有兩個偵測指令判斷要不要編譯 Opcode 。如果要,編譯器將使用 DynASM 將此 Opcode 轉換為機器碼,並執行此機器碼。

有趣的是,由於目前介面中編譯的程式碼有 MB 的限制 (也是可設定的),所以程式碼執行必須能夠在 JIT 和解釋程式碼之間無縫切換。

順便說一句,Benoit Jacquemont 在 php 的 JIT 上的這篇演講幫助我理解了這整件事。

我仍然不確定編譯部分什麼時候有效進行,但我想現在我真的不想知道。

所以你的效能收益可能不會很大

我希望現在大家都很清楚為什麼大多數php 應用程式不會因為使用即時編譯器而獲得很大的性能收益。這也是為什麼 Zeev 建議為你的應用程式分析和試驗不同的 JIT 配置是最好的方法。

如果您使用的是 PHP FPM,則通常會在多個請求之間共用已編譯的操作碼,但這仍然不能改變遊戲規則。

這是因為 JIT 優化了計算密集型的操作,而如今大多數 php 應用程式比其他任何東西都更受 I/O 約束。如果您無論如何都要存取磁碟或網絡,則處理操作是否已編譯則無關緊要。時間上將非常相似。

除非…

你正在做一些不受 I/O 約束的事情, 像是映像處理或機器學習。任何不接觸 I/O 的東西都將受益於 JIT 編譯器。

這也是為什麼現在人們說我們更願意用 PHP 寫原生功能而不是 C 寫的原因。如果仍然要編譯此功能,則開銷將毫無表現力。

有趣的時光成為一個 PHP 程式設計師…

相關教學推薦:《PHP教學

以上是了解一下PHP 8的 JIT 特性!的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
4 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
4 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
4 週前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

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

在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

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

解釋PHP中晚期靜態結合的概念。 解釋PHP中晚期靜態結合的概念。 Mar 21, 2025 pm 01:33 PM

文章討論了PHP 5.3中介紹的PHP中的晚期靜態結合(LSB),允許靜態方法的運行時間分辨率調用以更靈活的繼承。 LSB的實用應用和潛在的觸摸

框架安全功能:防止漏洞。 框架安全功能:防止漏洞。 Mar 28, 2025 pm 05:11 PM

文章討論了框架中的基本安全功能,以防止漏洞,包括輸入驗證,身份驗證和常規更新。

自定義/擴展框架:如何添加自定義功能。 自定義/擴展框架:如何添加自定義功能。 Mar 28, 2025 pm 05:12 PM

本文討論了將自定義功能添加到框架上,專注於理解體系結構,識別擴展點以及集成和調試的最佳實踐。

如何用PHP的cURL庫發送包含JSON數據的POST請求? 如何用PHP的cURL庫發送包含JSON數據的POST請求? Apr 01, 2025 pm 03:12 PM

使用PHP的cURL庫發送JSON數據在PHP開發中,經常需要與外部API進行交互,其中一種常見的方式是使用cURL庫發送POST�...

描述紮實的原則及其如何應用於PHP的開發。 描述紮實的原則及其如何應用於PHP的開發。 Apr 03, 2025 am 12:04 AM

SOLID原則在PHP開發中的應用包括:1.單一職責原則(SRP):每個類只負責一個功能。 2.開閉原則(OCP):通過擴展而非修改實現變化。 3.里氏替換原則(LSP):子類可替換基類而不影響程序正確性。 4.接口隔離原則(ISP):使用細粒度接口避免依賴不使用的方法。 5.依賴倒置原則(DIP):高低層次模塊都依賴於抽象,通過依賴注入實現。

ReactPHP的非阻塞特性究竟是什麼?如何處理其阻塞I/O操作? ReactPHP的非阻塞特性究竟是什麼?如何處理其阻塞I/O操作? Apr 01, 2025 pm 03:09 PM

深入解讀ReactPHP的非阻塞特性ReactPHP的一段官方介紹引起了不少開發者的疑問:“ReactPHPisnon-blockingbydefault....

See all articles