为了以后能开发PHP扩展,就一定要了解PHP的执行顺序。这篇文章就是为C开发PHP扩展做铺垫。

Web环境我们假设为Apache。在编译PHP的时候,为了能够让Apache支持PHP,我们会生成一个mod_php5.so的模块。Apache加载这个模块,在url访问.php文件的时候,就会转给mod_php5.so模块来处理。

这个就是我们常说的SAPI。英文名字是:Server Application Programming Interface。SAPI其实是一个统称,其下有 ISAPI,CLI SAPI,CGI等。有了它,就可以很容易的跟其他东西交互,比如APACHE,IIS,CGI等。

Apache启动后会将mod_pho5.so模块的hook handler注册进来,当Apache检测到访问的url是一个php文件时,这时候就会把控制权交给SAPI。进入到SAPI后,首先会执行sapi/apache/mod_php5.c 文件的php_init_handler函数,

这里摘录一段代码:

static void php_init_handler(server_rec *s, pool *p){

register_cleanup(p, NULL, (void (*)(void *))apache_php_module_shutdown_wrapper, (void (*)(void *))php_module_shutdown_for_exec);

if (!apache_php_initialized) {

apache_php_initialized = 1;

#ifdef ZTS         tsrm_startup(1, 1, 0, NULL);

#endif         sapi_startup(&apache_sapi_module);

php_apache_startup(&apache_sapi_module);

}

#if MODULE_MAGIC_NUMBER >= 19980527     {

TSRMLS_FETCH();

if (PG(expose_php)) {

ap_add_version_component("PHP/" PHP_VERSION);

}

}

#endif }

该函数主要调用两个函数:sapi_startup(&apache_sapi_module); php_apache_startup(&apache_sapi_module);

SAPI_API void sapi_startup(sapi_module_struct *sf){

sf->ini_entries = NULL;

sapi_module = *sf;

.................

sapi_globals_ctor(&sapi_globals);

................

virtual_cwd_startup(); /* Could use shutdown to free the main cwd but it would just slow it down for CGI */     ..................

reentrancy_startup();

}

sapi_startup创建一个 sapi_globals_struct结构体。sapi_globals_struct保存了Apache请求的基本信息,如服务器信息,Header,编码等。sapi_startup执行完毕后再执行php_apache_startup。

static int php_apache_startup(sapi_module_struct *sapi_module){

if (php_module_startup(sapi_module, &apache_module_entry, 1) == FAILURE) {

return FAILURE;

} else {

return SUCCESS;

}

}

php_module_startup 内容太多,这里介绍一下大致的作用:

1. 初始化zend_utility_functions 结构.这个结构是设置zend的函数指针,比如错误处理函数,输出函数,流操作函数等.

2. 设置环境变量.

3. 加载php.ini配置.

4. 加载php内置扩展.

5. 写日志.

6. 注册php内部函数集.

7. 调用 php_ini_register_extensions,加载所有外部扩展

8. 开启所有扩展

9. 一些清理操作.

重点说一下 3,4,7,8

加载php.ini配置

if (php_init_config(TSRMLS_C) == FAILURE) {

return FAILURE;

}

php_init_config函数会在这里检查所有php.ini配置,并且找到所有加载的模块,添加到php_extension_lists结构中。

加载php内置扩展

调用 zend_register_standard_ini_entries加载所有php的内置扩展,如array,mysql等。

调用 php_ini_register_extensions,加载所有外部扩展

main/php_ini.c

void php_ini_register_extensions(TSRMLS_D){

zend_llist_apply(&extension_lists.engine, php_load_zend_extension_cb TSRMLS_CC);

zend_llist_apply(&extension_lists.functions, php_load_php_extension_cb TSRMLS_CC);

zend_llist_destroy(&extension_lists.engine);

zend_llist_destroy(&extension_lists.functions);

}

zend_llist_apply函数遍历extension_lists 执行回调函数php_load_php_extension_cb

static void php_load_zend_extension_cb(void *arg TSRMLS_DC){

zend_load_extension(*((char **) arg));

}

该函数最后调用

if ((module_entry = zend_register_module_ex(module_entry TSRMLS_CC)) == NULL) {

DL_UNLOAD(handle);

return FAILURE;

}

将扩展信息放到 Hash表module_registry中,Zend/zend_API.c

if (zend_hash_add(&module_registry, lcname, name_len+1, (void *)module, sizeof(zend_module_entry), (void**)&module_ptr)==FAILURE) {

zend_error(E_CORE_WARNING, "Module \'%s\' already loaded", module->name);

efree(lcname);

return NULL;

}

最后,zend_startup_modules(TSRMLS_C); 对模块进行排序,并检测是否注册到module_registry HASH表里。zend_startup_extensions(); 执行extension->startup(extension);启动扩展。

一个php请求的执行过程,PHP程序执行的过程原理相关推荐

  1. 51单片机外部中断实验 设置中断优先级寄存器,当有外部中断0请求中断时,中断程序执行发光二极管程序,在此过程中,外部中断1也有中断请求,外部中断0的中断程序将被中断去执行外部中断1的中断程序(数码管

    设置中断优先级寄存器,当有外部中断0请求中断时,中断程序执行发光二极管程序,在此过程中,外部中断1也有中断请求,外部中断0的中断程序将被中断去执行外部中断1的中断程序(数码管加1显示程序). #inc ...

  2. Hello程序的转换处理过程及程序执行时的数据流动过程

    文章目录 1 Hello程序的转换处理过程 2 Hello程序执行时的数据流动过程 1 Hello程序的转换处理过程 经典的" hello.c "C-源程序: hello.c的AS ...

  3. android 启动app过程,应用程序进程启动过程

    原标题:应用程序进程启动过程 作者:慕涵盛华 链接:https://www.jianshu.com/p/b158615cc2ad 一.背景 首先注意的是:这里要说的是应用程序进程的启动过程,而不是应用 ...

  4. java语言执行过程_Java程序的运行过程(执行流程)分析

    万事知其然,要知其所以然,所以本节带大家来详细了解一下 Java 程序的执行过程.从<使用记事本编写运行Java程序>一节的案例可以看出,Java 程序的运行必须经过编写.编译和运行 3 ...

  5. C语言程序makefile编译过程

    C语言程序makefile编译过程 2022-0111 blog_010 makefile是一些大型linux下开发的项目经常会用到的编译脚本: 可以将makefile理解为类似shell一样的脚本语 ...

  6. tomcat 执行一个web请求的过程

    2019独角兽企业重金招聘Python工程师标准>>> Tomcat Server处理一个http请求的过程 假设来自客户的请求为: http://localhost:8080/ws ...

  7. zend解释php过程,Zend framework处理一个http请求的流程分析

    1, 首先是bootstrap过程,初始化程序里用到的资源 2, 创建一个Zend_Controller_Front实体,实现front controller模式,这个实体类会负责将http请求派遣到 ...

  8. Tomcat系列(6)——Tomcat处理一个HTTP请求的过程

    Tomcat的架构图   图三:Tomcat Server处理一个HTTP请求的过程 处理HTTP请求过程 假设来自客户的请求为:http://localhost:8080/test/index.js ...

  9. 小程序执行运行过程原理_分享 | 还在焦虑等成绩?成绩小助手了解下

    在咱们北林,期末考试后,我们往往不知道具体什么时候出成绩,以致我们隔三岔五地就要登一次系统,看看成绩是否更新,甚是麻烦. 为了缓解同学们反复查询的苦恼和焦虑的心情,我用python编写了一个小程序「成 ...

最新文章

  1. pandas基于条件判断更新dataframe中特定数据列数值内容的值(Conditionally updating values in specific pandas Dataframe )
  2. 基于JSP实现医院病历管理系统
  3. 使用HTML5画布实现的超棒javascript动画仪表板:gauge.js
  4. Spring Boot OAuth 2.0 客户端
  5. JBoss核心Java Web服务
  6. python 计量经济 35岁 工作_Python在计量经济与统计学中的应用
  7. Semaphore实现原理分析
  8. python-多线程共享内存
  9. Fragment控件初始化
  10. css行内样式的属性设置,css的外部样式的设置
  11. 如何设置.net控件SplitContainer平均分配
  12. 广告关闭php,电脑右下角的广告怎么去掉
  13. 计算机控制鼠标,键盘控制鼠标,详细教您如何使用键盘来控制鼠标
  14. 四季芬芳,花开不只在春天
  15. 毕业生如何应对职场挑战?分…
  16. 【IoT】NFC 应用技术全面解析
  17. Android实现垂直型的SeekBar
  18. 【Windows 10 驱动程序 已解决】重装 Win 10 系统后 电脑无法识别外接屏幕。投影:你的电脑不能投影到其他屏幕。请尝试重新安装驱动程序或使用其他视频卡。
  19. 视频去水印免费软件有哪些?视频去水印软件大分享
  20. IP地址转换 —— C/C++(来源PTA)

热门文章

  1. Java的设计模式 之 简单的工厂模式(一)
  2. Python之函数进阶
  3. 实验四+040+薛龚
  4. 一路风景---我期待的是师生关系
  5. avalon视频学习笔记(六)
  6. BZOJ2190 SDOI2008 仪仗队
  7. 在sqlServer中把数据导出为insert脚本
  8. Ubuntu8.04系列二-系统优化篇
  9. cordova与android通信_5:Cordova与原生交互--传值
  10. Python简易验证码生成程序