PHP模块开发学习心得2013-09-05最近在学习PHP模块开发相关的知识, 再看了dl()函数的流程之后, 对模块加载的处理流程做一个总结, 希望可以在PHP模块开发上帮助到大家.进入正题.PHP的代码架构

上图摘自Extending and Embedding PHP(Sams).从图中可以看出, PHP所有的部分都处在一个被称为TSRM的层中, TSRM层是负责线程安全管理的. 最底下的SAPI是对外提供服务的接口, 比如命令行的sapi为cli, php-fpm则是fastcgi的sapi, apache的模块方式也是一种sapi.中间是PHP内核和Zend 引擎. 从图中的文字可以看出, PHP内核负责请求管理/网络和文件操作, Zend内核则负责编译和执行/内存和资源的分配.在所有这些之上, 是扩展层, PHP中多数对外接口都是通过扩展层来提供的, 比如, standard, string等语言基础也被以扩展形式提供.扩展(以后称为模块)加载到PHP中的方式有两种: 静态编译, 动态链接.静态编译需要重新生成php的configure脚本, 这里不再赘述. 动态链接方式是将模块编译为一个.so文件, 然后动态的加载到php中.加载.so文件的方式有两种, 一种是将其写到php.ini文件中, 比如: extension=apc.so, 另外一种就是在代码中使用dl(‘xxx.so’).dl($library)函数的作用就是把一个模块加载进来, 使其内部提供的能力可用.dl()函数的源代码在PHP源代码根目录(简写为PHP_SRC_HOME)下, PHP_SRC_HOME/ext/standard/dl.c, 处理关键流程如下:PHP_FUNCTION(dl)
PHPAPI PHP_FUNCTION(dl){//...php_dl(filename, MODULE_TEMPORARY, return_value, 0 TSRMLS_CC);//... }php_dl
PHPAPI void php_dl(char *file, int type, zval *return_value, int start_now TSRMLS_DC){if (php_load_extension(file, type, start_now TSRMLS_CC) == FAILURE) { //...}php_load_extension
PHPAPI int php_load_extension(char *filename, int type, int start_now TSRMLS_DC) {//文件名解析相关//加载动态链接库handle = DL_LOAD(libpath);//加载错误处理//获取模块的get_module函数(重点, 模块初始入口)get_module = (zend_module_entry *(*)(void)) DL_FETCH_SYMBOL(handle, "get_module");//get_module函数获取错误处理//那个get_module()得到struct zend_module_entrymodule_entry = get_module();//...//注册模块(重点, 函数在这里被注册)if ((module_entry = zend_register_module_ex(module_entry TSRMLS_CC)) == NULL) {//错误处理}//模块启动(重点, PHP_MINIT_FUNCTION)if ((type == MODULE_TEMPORARY || start_now) && zend_startup_module_ex(module_entry TSRMLS_CC) == FAILURE) {//错误处理}//模块请求启动(重点, PHP_RINIT_FUNCTION)if ((type == MODULE_TEMPORARY || start_now) && module_entry->request_startup_func) {//错误处理}return SUCCESS;}