如何編譯php文件

發(fā)布時間:2024-03-08
php是解析型高級語言,事實上從zend內(nèi)核的角度來看php就是一個普通的c程序,它有main函數(shù),我們寫的php代碼是這個程序的輸入,然后經(jīng)過內(nèi)核的處理輸出結(jié)果,內(nèi)核將php代碼”翻譯”為c程序可識別的過程就是php的編譯。
c程序在編譯時將一行行代碼編譯為機(jī)器碼,每一個操作都認(rèn)為是一條機(jī)器指令,這些指令寫入到編譯后的二進(jìn)制程序中,執(zhí)行的時候?qū)⒍M(jìn)制程序load進(jìn)相應(yīng)的內(nèi)存區(qū)域(常量區(qū)、數(shù)據(jù)區(qū)、代碼區(qū))、分配運(yùn)行棧,然后從代碼區(qū)起始位置開始執(zhí)行,這是c程序編譯、執(zhí)行的簡單過程。
同樣,php的編譯與普通的c程序類似,只是php代碼沒有編譯成機(jī)器碼,而是解析成了若干條opcode數(shù)組,每條opcode就是c里面普通的struct,含義對應(yīng)c程序的機(jī)器指令,執(zhí)行的過程就是引擎依次執(zhí)行opcode,比如我們在php里定義一個變量:$a = 123;,最終到內(nèi)核里執(zhí)行就是malloc一塊內(nèi)存,然后把值寫進(jìn)去。
在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; //運(yùn)行時緩存數(shù)組大小 int cache_size; //運(yùn)行時緩存,主要用于緩存一些znode_op以便于快速獲取數(shù)據(jù),后面單獨介紹這個機(jī)制 void run_time_cache; void *reserved[zend_max_reserved_resources]; };opcode指令:即php代碼具體對應(yīng)的處理動作,與二進(jìn)制程序中的代碼段對應(yīng)
字面量存儲:php代碼中定義的一些變量初始值、調(diào)用的函數(shù)名稱、類名稱、常量名稱等等稱之為字面量,這些值用于執(zhí)行時初始化變量、函數(shù)調(diào)用等等
變量分配情況:與字面量類似,這里指的是當(dāng)前opcodes定義了多少變量、臨時變量,每個變量都有一個對應(yīng)的編號,執(zhí)行初始化按照總的數(shù)目一次性分配zval,使用時也完全按照編號索引,而不是根據(jù)變量名索引
從php代碼到opcode是怎么實現(xiàn)的?
最容易想到的方式就是正則匹配,當(dāng)然過程沒有這么簡單。php編譯過程包括詞法分析、語法分析,使用re2c、bison完成,舊的php版本直接生成了opcode,php7新增了抽象語法樹(ast),在語法分析階段生成ast,然后再生成opcode數(shù)組
上一個:一體式電磁流量計價格,分體式電磁流量計價格表,電磁流量表價格
下一個:win10鍵盤鎖住了怎么辦(win10鍵盤鎖住了怎么辦恢復(fù))

提高PLC程序運(yùn)行速度的編程方法
淺析伺服電機(jī)慣量問題
電子信息專業(yè)就業(yè)方向
騰訊云服務(wù)器網(wǎng)站是哪個文件夾
惠普怎么掃描到電腦上面(惠普掃描怎么掃描)
申請強(qiáng)制執(zhí)行的期限是幾年
鋼筋銹蝕測定儀用途
20年電腦主流配置(2030年的電腦)
文心蘭的栽培技術(shù)與管理 養(yǎng)好芙蓉花注意點
筆記本電腦i5和i7區(qū)別,筆記本電腦i5和i7有什么區(qū)別
十八禁 网站在线观看免费视频_2020av天堂网_一 级 黄 色 片免费网站_绝顶高潮合集Videos