溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

編譯php文件的方法

發(fā)布時間:2020-08-25 14:23:23 來源:億速云 閱讀:155 作者:小新 欄目:編程語言

這篇文章將為大家詳細講解有關(guān)編譯php文件的方法,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

PHP是解析型高級語言,事實上從Zend內(nèi)核的角度來看PHP就是一個普通的C程序,它有main函數(shù),我們寫的PHP代碼是這個程序的輸入,然后經(jīng)過內(nèi)核的處理輸出結(jié)果,內(nèi)核將PHP代碼"翻譯"為C程序可識別的過程就是PHP的編譯。

推薦課程:PHP教程。

編譯php文件的方法

C程序在編譯時將一行行代碼編譯為機器碼,每一個操作都認為是一條機器指令,這些指令寫入到編譯后的二進制程序中,執(zhí)行的時候?qū)⒍M制程序load進相應(yīng)的內(nèi)存區(qū)域(常量區(qū)、數(shù)據(jù)區(qū)、代碼區(qū))、分配運行棧,然后從代碼區(qū)起始位置開始執(zhí)行,這是C程序編譯、執(zhí)行的簡單過程。

同樣,PHP的編譯與普通的C程序類似,只是PHP代碼沒有編譯成機器碼,而是解析成了若干條opcode數(shù)組,每條opcode就是C里面普通的struct,含義對應(yīng)C程序的機器指令,執(zhí)行的過程就是引擎依次執(zhí)行opcode,比如我們在PHP里定義一個變量:$a = 123;,最終到內(nèi)核里執(zhí)行就是malloc一塊內(nèi)存,然后把值寫進去。

在zend_compile.h文件中,opcode結(jié)構(gòu):

struct _zend_op {
    const void *handler; //對應(yīng)執(zhí)行的C語言function,即每條opcode都有一個C function處理
    znode_op op1; //操作數(shù)1
    znode_op op2; //操作數(shù)2
    znode_op result; //返回值
    uint32_t extended_value; 
    uint32_t lineno;
    zend_uchar opcode;  //opcode指令
    zend_uchar op1_type; //操作數(shù)1類型
    zend_uchar op2_type; //操作數(shù)2類型
    zend_uchar result_type; //返回值類型
};


所以PHP的解析過程任務(wù)就是將PHP代碼(通過詞法分析re2c,語法分析bison)轉(zhuǎn)化為opcode數(shù)組,代碼里的所有信息都保存在opcode中,然后將opcode數(shù)組交給zend引擎執(zhí)行,opcode就是內(nèi)核具體執(zhí)行的命令,比如賦值、加減操作、函數(shù)調(diào)用等,每一條opcode都對應(yīng)一個處理handle,這些handler是提前定義好的C函數(shù)。

struct _zend_op_array {
    //common是普通函數(shù)或類成員方法對應(yīng)的opcodes快速訪問時使用的字段
    /* Common elements */
    zend_uchar type;
    zend_uchar arg_flags[3]; /* bitset of arg_info.pass_by_reference */
    uint32_t fn_flags;
    zend_string *function_name;
    zend_class_entry *scope;
    zend_function *prototype;
    uint32_t num_args;
    uint32_t required_num_args;
    zend_arg_info *arg_info;
    /* END of common elements */

    uint32_t *refcount;

    uint32_t last;
     //opcode指令數(shù)組
    zend_op *opcodes;

    //PHP代碼里定義的變量數(shù):op_type為IS_CV的變量,不含IS_TMP_VAR、IS_VAR的
    //編譯前此值為0,然后發(fā)現(xiàn)一個新變量這個值就加1
    int last_var;
    //臨時變量數(shù):op_type為IS_TMP_VAR、IS_VAR的變量
    uint32_t T;
    //PHP變量名數(shù)組
    zend_string **vars;//這個數(shù)組在ast編譯期間配合last_var用來確定各個變量的編號,非常重要的一步操作

    int last_live_range;
    int last_try_catch;
    zend_live_range *live_range;
    zend_try_catch_element *try_catch_array;

    //靜態(tài)變量符號表:通過static聲明的
    /* static variables support */
    HashTable *static_variables;

    zend_string *filename;
    uint32_t line_start;
    uint32_t line_end;
    zend_string *doc_comment;
    uint32_t early_binding; /* the linked list of delayed declarations */

    //字面量數(shù)量
    int last_literal;
    //字面量(常量)數(shù)組,這些都是在PHP代碼定義的一些值
    zval *literals;

    //運行時緩存數(shù)組大小
    int  cache_size;
    //運行時緩存,主要用于緩存一些znode_op以便于快速獲取數(shù)據(jù),后面單獨介紹這個機制
    void **run_time_cache;

    void *reserved[ZEND_MAX_RESERVED_RESOURCES];
};

opcode指令:即PHP代碼具體對應(yīng)的處理動作,與二進制程序中的代碼段對應(yīng)
字面量存儲:PHP代碼中定義的一些變量初始值、調(diào)用的函數(shù)名稱、類名稱、常量名稱等等稱之為字面量,這些值用于執(zhí)行時初始化變量、函數(shù)調(diào)用等等
變量分配情況:與字面量類似,這里指的是當前opcodes定義了多少變量、臨時變量,每個變量都有一個對應(yīng)的編號,執(zhí)行初始化按照總的數(shù)目一次性分配zval,使用時也完全按照編號索引,而不是根據(jù)變量名索引

從PHP代碼到opcode是怎么實現(xiàn)的?

最容易想到的方式就是正則匹配,當然過程沒有這么簡單。PHP編譯過程包括詞法分析、語法分析,使用re2c、bison完成,舊的PHP版本直接生成了opcode,PHP7新增了抽象語法樹(AST),在語法分析階段生成AST,然后再生成opcode數(shù)組

關(guān)于編譯php文件的方法就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI