您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關PHP自定義擴展之生命周期的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
php_request_startup
這個階段和php_module_startup
差不多,都是初始化工作,比php_module_startup
簡單很多,可以自己看下,重點來看下執行階段
php_execute_script
用gdb看看調用棧,gdb ./php
在php_execute_script
打斷點,執行,在看下調用棧,
b php_execute_script (gdb) r test.php bt #0 php_execute_script (primary_file=0x7fffffffe240) at /www/test/php/php-7.4.3/main/main.c:2541 #1 0x00000000008bbd85 in do_cli (argc=2, argv=0x1425af0) at /www/test/php/php-7.4.3/sapi/cli/php_cli.c:961 #2 0x00000000008bcd2d in main (argc=2, argv=0x1425af0) at /www/test/php/php-7.4.3/sapi/cli/php_cli.c:1356
在調用棧可以清楚看到執行流程,現在到/cli/php_cli.c文件看看做了哪些事情,
int c; zend_file_handle file_handle; int behavior = PHP_MODE_STANDARD; char *reflection_what = NULL; volatile int request_started = 0; volatile int exit_status = 0; char *php_optarg = NULL, *orig_optarg = NULL; int php_optind = 1, orig_optind = 1; char *exec_direct=NULL, *exec_run=NULL, *exec_begin=NULL, *exec_end=NULL; char *arg_free=NULL, **arg_excp=&arg_free; char *script_file=NULL, *translated_path = NULL; int interactive=0; int lineno = 0; const char *param_error=NULL; int hide_argv = 0; zend_try { CG(in_compilation) = 0; /* not initialized but needed for several options */ while ((c = php_getopt(argc, argv, OPTIONS, &php_optarg, &php_optind, 0, 2)) != -1) { switch (c) {
初始化變量,解析命令到/main/main.c文件看看真正的執行階段
PHPAPI int php_execute_script(zend_file_handle *primary_file) { zend_file_handle *prepend_file_p, *append_file_p; zend_file_handle prepend_file = {{0}, NULL, NULL, 0, 0}, append_file = {{0}, NULL, NULL, 0, 0}; #if HAVE_BROKEN_GETCWD volatile int old_cwd_fd = -1; #else char *old_cwd; ALLOCA_FLAG(use_heap) #endif int retval = 0;
加載要執行的php文件,通過zend_compile_file
進行詞法分析 語法分析,生成AST
,編譯成op_array
,也就是指令集,
我們看下指令集,
b zend_execute c bt (gdb) p *op_array $1 = {type = 2 '\002', arg_flags = "\000\000", fn_flags = 37748736, function_name = 0x0, scope = 0x0, prototype = 0x0, num_args = 0, required_num_args = 0, arg_info = 0x0, cache_size = 16, last_var = 2, T = 4, last = 13, opcodes = 0x7ffff5e8b000, run_time_cache__ptr = 0x0, static_variables_ptr__ptr = 0x7ffff5e78358, static_variables = 0x0, vars = 0x7ffff5e790f0, refcount = 0x7ffff5e85000, last_live_range = 0, last_try_catch = 0, live_range = 0x0, try_catch_array = 0x0, filename = 0x7ffff5e583c0, line_start = 1, line_end = 12, doc_comment = 0x0, last_literal = 5, literals = 0x7ffff5e8b1a0, reserved = {0x0, 0x0, 0x0, 0x0, 0x0, 0x0}}
然后放到zend虛擬機上執行zend_execute(op_array, retval);
這這里可以看到真正執行的是這個zend_execute_ex(execute_data);
,它是一個函數指針,我們可以更換它,知道這個,那么我們在寫擴展時可以重新寫個zend_execute_ex
函數替換php
默認的,我們自己的就可以做很多事情,像攔截php函數,做性能監控。
請求關閉階段php_request_shutdown
,模塊關閉階段php_module_shutdown
這兩個階段主要是做變量銷毀,現在我們知道了一個自定義擴展在生命周期里的怎么執行的。
關于“PHP自定義擴展之生命周期的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。