## 3.1 PHP代碼的編譯
PHP是解析型高級語言,事實上從Zend內核的角度來看PHP就是一個普通的C程序,它有main函數,我們寫的PHP代碼是這個程序的輸入,然后經過內核的處理輸出結果,內核將PHP代碼"翻譯"為C程序可識別的過程就是PHP的編譯。
那么這個"翻譯"過程具體都有哪些操作呢?
C程序在編譯時將一行行代碼編譯為機器碼,每一個操作都認為是一條機器指令,這些指令寫入到編譯后的二進制程序中,執行的時候將二進制程序load進相應的內存區域(常量區、數據區、代碼區)、分配運行棧,然后從代碼區起始位置開始執行,這是C程序編譯、執行的簡單過程。
同樣,PHP的編譯與普通的C程序類似,只是PHP代碼沒有編譯成機器碼,而是解析成了若干條opcode數組,每條opcode就是C里面普通的struct,含義對應C程序的機器指令,執行的過程就是引擎依次執行opcode,比如我們在PHP里定義一個變量:`$a = 123;`,最終到內核里執行就是malloc一塊內存,然后把值寫進去。
所以PHP的解析過程任務就是將PHP代碼轉化為opcode數組,代碼里的所有信息都保存在opcode中,然后將opcode數組交給zend引擎執行,opcode就是內核具體執行的命令,比如賦值、加減操作、函數調用等,每一條opcode都對應一個處理handle,這些handler是提前定義好的C函數。
從PHP代碼到opcode是怎么實現的?最容易想到的方式就是正則匹配,當然過程沒有這么簡單。PHP編譯過程包括詞法分析、語法分析,使用re2c、bison完成,舊的PHP版本直接生成了opcode,PHP7新增了抽象語法樹(AST),在語法分析階段生成AST,然后再生成opcode數組。

PHP編譯階段的基本過程如下圖:

后面兩個小節將看下 __PHP代碼->AST->Opcodes__ 的具體編譯過程。
- 目錄
- 第1章 PHP基本架構
- 1.1 PHP簡介
- 1.2 PHP7的改進
- 1.3 FPM
- 1.4 PHP執行的幾個階段
- 第2章 變量
- 2.1 變量的內部實現
- 2.2 數組
- 2.3 靜態變量
- 2.4 全局變量
- 2.5 常量
- 3.1 PHP代碼的編譯
- 3.1.1 詞法解析、語法解析
- 3.1.2 抽象語法樹編譯流程
- 第3章 Zend虛擬機
- 3.2.1 內部函數
- 3.2.2 用戶函數的實現
- 3.3 Zend引擎執行流程
- 3.3.1 基本結構
- 3.2 函數實現
- 3.3.2 執行流程
- 3.3.3 函數的執行流程
- 3.3.4 全局execute_data和opline
- 3.4 面向對象實現
- 3.4.1 類
- 3.4.2 對象
- 3.4.3 繼承
- 3.4.4 動態屬性
- 3.4.5 魔術方法
- 3.4.6 類的自動加載
- 3.5 運行時緩存
- 3.6 Opcache
- 3.6.1 opcode緩存
- 3.6.2 opcode優化
- 3.6.3 JIT
- 第4章 PHP基礎語法實現
- 4.1 類型轉換
- 4.2 選擇結構
- 4.3 循環結構
- 4.4 中斷及跳轉
- 4.5 include/require
- 4.6 異常處理
- 第5章 內存管理
- 5.1 Zend內存池
- 5.2 垃圾回收
- 第6章 線程安全
- 6.2 線程安全資源管理器
- 第7章 擴展開發
- 7.1 概述
- 6.1 什么是線程安全
- 7.2 擴展的實現原理
- 7.3 擴展的構成及編譯
- 7.4 鉤子函數
- 7.5 運行時配置
- 7.6 函數
- 7.7 zval的操作
- 7.8 常量
- 7.9 面向對象
- 7.9.1 內部類注冊
- 7.9.2 定義成員屬性
- 7.9.3 定義成員方法
- 7.9.4 定義常量
- 7.9.5 類的實例化
- 7.10 資源類型
- 7.11 經典擴展解析
- 7.8.1 Yaf
- 7.8.2 Redis
- 第8章 命名空間
- 8.2 命名空間的定義
- 8.2.1 定義語法
- 8.2.2 內部實現
- 8.3 命名空間的使用
- 8.3.1 基本用法
- 8.3.2 use導入
- 8.3.3 動態用法
- 附錄
- 附錄1:break/continue按標簽中斷語法實現
- 附錄2:defer推遲函數調用語法的實現
- 8.1 概述