首頁 後端開發 PHP問題 深入理解PHP原理之PHP與WEB伺服器交互

深入理解PHP原理之PHP與WEB伺服器交互

Aug 19, 2019 pm 03:27 PM
php

大家都知道,PHP需要在具體的WEB伺服器中才能運行,例如Nginx、Apache等,但是PHP是怎麼啟動,又是怎樣在伺服器中運行,然後兩者又是怎樣進行交互的呢? 

1.WEB伺服器呼叫PHP介面 

#  以Apache伺服器為例,我們來看看伺服器是如何啟動PHP,並且呼叫PHP中的方法。 Apache伺服器啟動並執行PHP時,一般是透過mod_php7模組的形式整合(如果是php5.*版本,就是mod_php5模組,模組後綴名稱依php版本而定), mod_php7的結構如下(源碼路徑為php/sapi/apache2handler/mod_php7.c):

AP_MODULE_DECLARE_DATA module php7_module = {
    STANDARD20_MODULE_STUFF,/* 宏,包括版本,版本,模块索引,模块名,下个模块指针等信息 */
    create_php_config,      /* create per-directory config structure */
    merge_php_config,       /* merge per-directory config structures */
    NULL,                   /* create per-server config structure */
    NULL,                   /* merge per-server config structures */
    php_dir_cmds,           /* 模块定义的所有指令 */
    php_ap2_register_hook   /* register hooks */
};
登入後複製

當Apache需要呼叫PHP中的方法時,只需要將此請求透過mod_php7模組傳達給PHP,PHP層處理完後將資料回傳給Apache,整個過程就結束了(補充:Apache伺服器啟動PHP時,其實有兩種加載方式,一種為靜態加載,一種為動態加載,剛才討論的mod_php5模組加載方式可以理解為靜態加載,也就是需要重新啟動Apache伺服器,才能將PHP加載進去;動態加載不需要重啟伺服器,只需要透過發送訊號的方式將PHP固定的模組載入到伺服器,以達到PHP啟動的目的,但是在進行動態載入之前,需要將載入模組編譯成動態連結庫,然後將其配置到伺服器的設定檔中)。上面已經給出Apache在PHP中的model結構,下面給出Apache伺服器中對應的module結構,如下(該原始碼在Apache中,下同):

struct module_struct {
    int version;
    int minor_version;
    int module_index;
    const char *name;
    void *dynamic_load_handle;
    struct module_struct *next;
    unsigned long magic;
    void (*rewrite_args) (process_rec *process);
    void *(*create_dir_config) (apr_pool_t *p, char *dir);
    void *(*merge_dir_config) (apr_pool_t *p, void *base_conf, void *new_conf);
    void *(*create_server_config) (apr_pool_t *p, server_rec *s);
    void *(*merge_server_config) (apr_pool_t *p, void *base_conf, void *new_conf);
    const command_rec *cmds;
    void (*register_hooks) (apr_pool_t *p);
}
登入後複製

可以看得出php7_module和module_struct還是有很大不同,不過如果看到php7_module.STANDARD20_MODULE_STUFF這個巨集的定義方式,你可能會覺得這兩個結構體很像,其實這個巨集定義了module_struct中的前8個參數,定義如下:

#define STANDARD20_MODULE_STUFF MODULE_MAGIC_NUMBER_MAJOR, \
    MODULE_MAGIC_NUMBER_MINOR, \
    -1, \
    __FILE__, \
    NULL, \
    NULL, \
    MODULE_MAGIC_COOKIE, \
    NULL /* rewrite args spot */
登入後複製

然後php7_module.php_dir_cmds定義了模組的所有指令集合,具體定義內容如下(程式碼路徑為php/sapi/apache2handler/apache_config.c):

const command_rec php_dir_cmds[] =
{
    AP_INIT_TAKE2("php_value", php_apache_value_handler, NULL, OR_OPTIONS, "PHP Value Modifier"),
    AP_INIT_TAKE2("php_flag", php_apache_flag_handler, NULL, OR_OPTIONS, "PHP Flag Modifier"),
    AP_INIT_TAKE2("php_admin_value", php_apache_admin_value_handler, NULL, ACCESS_CONF|RSRC_CONF, "PHP Value Modifier (Admin)
    "),
    AP_INIT_TAKE2("php_admin_flag", php_apache_admin_flag_handler, NULL, ACCESS_CONF|RSRC_CONF, "PHP Flag Modifier (Admin)"),
    AP_INIT_TAKE1("PHPINIDir", php_apache_phpini_set, NULL, RSRC_CONF, "Directory containing the php.ini file"),
    {NULL}
};
登入後複製

也就是說,PHP層只給Apache提供了上述5個指令,每個指令的實作原始碼也在apache_config.c檔中,最後就剩下php7_module.php_ap2_register_hook了,它定義的內容如下(程式碼路徑為php/sapi/apache2handler/mod_php7.c) :

void php_ap2_register_hook(apr_pool_t *p)
{
    ap_hook_pre_config(php_pre_config, NULL, NULL, APR_HOOK_MIDDLE);
    ap_hook_post_config(php_apache_server_startup, NULL, NULL, APR_HOOK_MIDDLE);
    ap_hook_handler(php_handler, NULL, NULL, APR_HOOK_MIDDLE);
#ifdef ZEND_SIGNALS
    ap_hook_child_init(zend_signal_init, NULL, NULL, APR_HOOK_MIDDLE);
#endif
    ap_hook_child_init(php_apache_child_init, NULL, NULL, APR_HOOK_MIDDLE);
}
登入後複製

php7_module.php_ap2_register_hook函數包含4個鉤子和對應的處理函數,pre_config,pre_config、post_config和child_init是啟動鉤子,它們是在伺服器啟動時調用,handler鉤子是請求掛鉤,它是在伺服器請求是調用,透過這些鉤子,就可以透過Apache伺服器啟動PHP。 

   將到這裡,想必大家已經知道WEB伺服器是如何啟動PHP,並且呼叫PHP中的方法了哈,下面再給大家講講PHP是如何呼叫WEB伺服器介面的。

2.PHP呼叫WEB伺服器介面 

  在講述這個問題前,我們要先了解什麼是SAPI。 SAPI其實是與伺服器抽象層之間遵守的共同約定,可以這麼簡單理解,當PHP需要呼叫伺服器中的方法,例如清除緩存,但是清除快取的實現方法是在伺服器中實現,PHP層根本就不知道怎麼呼叫伺服器中的該方法,怎麼辦?這時雙方需要進行約定,然後伺服器提供一套約定後的介面給PHP,我們把這些與伺服器抽象層之間遵守的共同約定稱為SAPI介面。

  問題來了,對於伺服器Apache,我們可以提供一套SAPI,但是如果下次又來個其它的伺服器,或者其它的“第三方”,那麼我們是不是也要給他們提供一套單獨的SAPI呢?我們聰明的PHP開發者肯定想到了這一點,即對所有的「第三方」提供一套通用的SAPI接口,但是你可以會問,如果新的「第三方」需要的接口,你的通用SAPI不支持,那怎麼辦呢,我的理解是將新的功能添加到PHP的通用SAPI接口中,僅僅是個人見解哈,通用SAPI結構如下(源碼路徑: php/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);   // 关闭函数
    int (*activate)(TSRMLS_D);           // 激活
    int (*deactivate)(TSRMLS_D);         // 停用
    void (*flush)(void *server_context); // flush
    char *(*read_cookies)(TSRMLS_D);     //read Cookies
    //...
};
登入後複製

該結構體變數較多,就不一一列舉,簡要說明一下裡面的變數:startup函數是當SAPI初始化時會被調用,shutdown函數是用來釋放SAPI的資料結構和記憶體等,read_cookie是在SAPI啟動時被調用,然後將此函數取得的值賦值給SG(request_info).cookie_data。那麼對於PHP提供的通用SAPI,Apache伺服器又是怎麼客製自己的介面呢?具體結構如下(源碼路徑為php/sapi/apache2handler/sapi_apache2.c):

static sapi_module_struct apache2_sapi_module = {
    "apache2handler",
    "Apache 2.0 Handler",

    php_apache2_startup,            /* startup */
    php_module_shutdown_wrapper,    /* shutdown */

    NULL,                           /* activate */
    NULL,                           /* deactivate */

    php_apache_sapi_ub_write,       /* unbuffered write */
    php_apache_sapi_flush,          /* flush */
    php_apache_sapi_get_stat,       /* get uid */
    php_apache_sapi_getenv,         /* getenv */

    php_error,                      /* error handler */

    php_apache_sapi_header_handler, /* header handler */
    php_apache_sapi_send_headers,   /* send headers handler */
    NULL,                           /* send header handler */

    php_apache_sapi_read_post,      /* read POST data */
    php_apache_sapi_read_cookies,   /* read Cookies */

    php_apache_sapi_register_variables,
    php_apache_sapi_log_message,        /* Log message */
    php_apache_sapi_get_request_time,   /* Request Time */
    NULL,                               /* Child Terminate */

    STANDARD_SAPI_MODULE_PROPERTIES
};
登入後複製

上述原始碼目錄php/sapi/apache2handler/中,目錄php/sapi下面放的都是透過SAPI呼叫的「第三方”,該目錄結構如下圖所示,目錄php/sapi/apache2handler中都是與PHP交互的接口,sapi_apache2.c是PHP與Apache約定的SAPI接口文件。 

深入理解PHP原理之PHP與WEB伺服器交互

#

看到这里,大家应该基本清楚PHP层是怎样调用服务器层的接口,为了巩固上面的知识,下面举个栗子,即在Apache服务器环境下读取cookie:

SG(request_info).cookie_data = sapi_module.read_cookies(TSRMLS_C);
登入後複製

 对于任意一个服务器在加载时,我们都会指定sapi_module,Apache的sapi_module是apache2_sapi_module,它的read_cookies方法的是php_apache_sapi_read_cookies函数,这样就实现PHP层调用Apache的接口,是不是很简单呢:)

更多相关问题可以访问PHP中文网相关教程:https://www.php.cn/course/list/29/type/2.html

以上是深入理解PHP原理之PHP與WEB伺服器交互的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.聊天命令以及如何使用它們
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

我後悔之前不知道的 7 個 PHP 函數 我後悔之前不知道的 7 個 PHP 函數 Nov 13, 2024 am 09:42 AM

如果您是經驗豐富的PHP 開發人員,您可能會感覺您已經在那裡並且已經完成了。操作

您如何在PHP中解析和處理HTML/XML? 您如何在PHP中解析和處理HTML/XML? Feb 07, 2025 am 11:57 AM

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

See all articles