程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> PHP模塊開發學習心得

PHP模塊開發學習心得

編輯:PHP綜合

最近在學習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_entry  
    module_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;  
}

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved