摘要:前言函數默認構建目標為,相關代碼在目錄下,文件中能夠找到入口函數,大概流程如下命令行參數處理初始化清理工作語言系統編程常用手法,通過中聲明函數指針類型的字段來實現類似面向對象中抽象類的概念,在文件中可以找到該結構體的定義,這里只列出部分
前言 php cli main 函數
configure & make 默認構建目標為 php-cli,相關代碼在 sapi/cli 目錄下,php_cli.c 文件中能夠找到 main(入口)函數,大概流程如下:
命令行參數處理
cli_sapi_module 初始化
sapi_module->startup
do_cli or do_cli_server
清理工作
sapi_module_structC 語言系統編程常用手法,通過 struct 中聲明 函數指針 類型的字段來實現類似面向對象中抽象類的概念,在 main/SAPI.h 文件中可以找到該結構體的定義,這里只列出部分字段(下同):
struct _sapi_module_struct { char *name; char *pretty_name; int (*startup)(struct _sapi_module_struct *sapi_module); int (*shutdown)(struct _sapi_module_struct *sapi_module); ... char *ini_entries; const zend_function_entry *additional_functions; unsigned int (*input_filter_init)(void); }cli_sapi_module
cli_sapi_module 是一個靜態全局變量,定義在 php_cli.c 中,你可以將它理解成是 sapi_module_struct "類" 的一個 "實例",結構體中 "掛載" 了 cli 特定的實現函數:
/* {{{ sapi_module_struct cli_sapi_module */ static sapi_module_struct cli_sapi_module = { "cli", /* name */ "Command Line Interface", /* pretty name */ ... php_cli_startup, /* startup */ php_module_shutdown_wrapper, /* shutdown */ ... STANDARD_SAPI_MODULE_PROPERTIES };do_cli
do_cli 函數定義在 php_cli.c 文件中,大致流程如下:
根據命令行選項 確定 behavior(解釋器行為)
根據 behavior 執行相應的動作
通過執行 sapi/cli/php --help 可以查看所有的 php-cli 命令行選項,我們通過幾個簡單的選項來分析解釋器執行流程
PHP_MODE_CLI_DIRECT該模式下,php-cli 會執行解釋執行通過命令行參數傳遞的 code
case PHP_MODE_CLI_DIRECT: cli_register_file_handles(); if (zend_eval_string_ex(exec_direct, NULL, "Command line code", 1) == FAILURE) { exit_status=254; } break;
追蹤 zend_eval_string_ex 的函數調用,定位到 zend_execute_API.c 文件中 zend_eval_stringl 函數,代碼邏輯已經很清楚了:先調用 zend_compile_string 函數編譯代碼生成字節碼 new_op_array,再調用 zend_execute 函數執行生成的字節碼
ZEND_API int zend_eval_stringl(char *str, size_t str_len, zval *retval_ptr, char *string_name) { ... original_compiler_options = CG(compiler_options); CG(compiler_options) = ZEND_COMPILE_DEFAULT_FOR_EVAL; new_op_array = zend_compile_string(&pv, string_name); CG(compiler_options) = original_compiler_options; if (new_op_array) { zend_try { ZVAL_UNDEF(&local_retval); zend_execute(new_op_array, &local_retval); } zend_catch { destroy_op_array(new_op_array); efree_size(new_op_array, sizeof(zend_op_array)); zend_bailout(); } zend_end_try(); ... } else { retval = FAILURE; } zval_dtor(&pv); return retval; }
zend_compile_string 屬于語法分析內容,參考 PHP-7.1 源代碼學習: 語法分析,這里做個簡要介紹
compile_string通過搜索源代碼可以發現 zend_compile_string 最終調用 compile_string
zend_op_array *compile_string(zval *source_string, char *filename) { zend_lex_state original_lex_state; zend_op_array *op_array = NULL; zval tmp; if (Z_STRLEN_P(source_string)==0) { return NULL; } ZVAL_DUP(&tmp, source_string); convert_to_string(&tmp); source_string = &tmp; zend_save_lexical_state(&original_lex_state); if (zend_prepare_string_for_scanning(source_string, filename) == SUCCESS) { BEGIN(ST_IN_SCRIPTING); op_array = zend_compile(ZEND_EVAL_CODE); } zend_restore_lexical_state(&original_lex_state); zval_dtor(&tmp); return op_array; }總結
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/22325.html
摘要:前言本文通過分析這個語句的編譯和執行來窺探解釋執行邏輯準備參考之前的系列文章,在環境下下載,編譯源代碼將代碼導入中編輯運行選項,增加運行參數設置斷點開始調試是一個測試腳本,放在目錄下,中只包含一條簡單的賦值語句調用堆棧參考之前的系列文章 前言 本文通過分析 $a=1 這個 PHP 語句的編譯和執行來窺探 php-cli 解釋執行邏輯 準備 參考之前的系列文章,在 ubuntu 環境下...
摘要:配置例如文件的目錄關系如下打開文件,修改的值為項目文件的相對路徑,這個路徑會映射到容器的目錄上,注意任何修改配置文件都需要執行容器名字重新構建相關容器。 背景 每次更換一個工作用的電腦,都需要很長的時間來搭建本地的運行環境。有時候,為了安裝一個擴展會折騰半天,而且可能需要安裝多個版本的 php。有人說,集成運行環境不是很好嘛,一次搞定所有的要求。是的,對于新手來說,使用集成包就足夠了。...
摘要:前言字節碼生成編譯的代碼主要集中在,文件中包含大量的函數,基本上一個函數對應語法規則文件一個非終結符,函數是所有函數的入口數據結構結構體是字節碼抽象結構體并沒有像名字那樣簡單,它包含了大量的字段供虛擬機在運行時使用一如既往的簡單,直觀,相比 前言 字節碼生成(編譯)的代碼主要集中在 zend_compile.c ,文件中包含大量的 zend_compile_xxx 函數,基本上一個函數...
摘要:開場白作為一個技術團隊的,你是如何保證成員的開發環境達到公司的標準,或者是你定制的最低要求的如果你的回答是差不多就行了,有問題再說,那么,你已經在給自己挖坑了。好的,成員們開始構建你定制的開發環境了。 開場白 作為一個技術團隊的Leader,你是如何保證成員的開發環境達到公司的標準,或者是你定制的最低要求的?如果你的回答是:差不多就行了,有問題再說,那么,你已經在給自己挖坑了。 同事A...
閱讀 1972·2021-11-23 10:03
閱讀 4130·2021-11-22 09:34
閱讀 2466·2021-10-08 10:05
閱讀 2247·2019-08-30 15:53
閱讀 1686·2019-08-30 13:56
閱讀 1149·2019-08-29 16:52
閱讀 1102·2019-08-26 13:31
閱讀 3346·2019-08-26 11:45