PHP內(nèi)核探索 —— 再次探討SAPI
在PHP的生命周期的各個(gè)階段,一些與服務(wù)相關(guān)的操作都是通過(guò)SAPI接口實(shí)現(xiàn)。 這些內(nèi)置實(shí)現(xiàn)的物理位置在PHP源碼的SAPI目錄。這個(gè)目錄存放了PHP對(duì)各個(gè)服務(wù)器抽象層的代碼, 例如命令行程序的實(shí)現(xiàn),Apache的mod_php5模塊實(shí)現(xiàn)以及fastcgi的實(shí)現(xiàn)等等。
在各個(gè)服務(wù)器抽象層之間遵守著相同的約定,這里我們稱之為SAPI接口。 每個(gè)SAPI實(shí)現(xiàn)都是一個(gè)_sapi_module_struct結(jié)構(gòu)體變量。(SAPI接口)。 在PHP的源碼中,當(dāng)需要調(diào)用服務(wù)器相關(guān)信息時(shí),全部通過(guò)SAPI接口中對(duì)應(yīng)方法調(diào)用實(shí)現(xiàn), 而這對(duì)應(yīng)的方法在各個(gè)服務(wù)器抽象層實(shí)現(xiàn)時(shí)都會(huì)有各自的實(shí)現(xiàn)。
下面是為SAPI的簡(jiǎn)單示意圖:
以cgi模式和apache2服務(wù)器為例,它們的啟動(dòng)方法如下:
cgi_sapi_module.startup(&cgi_sapi_module) // cgi模式 cgi/cgi_main.c文件apache2_sapi_module.startup(&apache2_sapi_module); // apache2服務(wù)器 apache2handler/sapi_apache2.c文件
這里的cgi_sapi_module是sapi_module_struct結(jié)構(gòu)體的靜態(tài)變量。 它的startup方法指向php_cgi_startup函數(shù)指針。在這個(gè)結(jié)構(gòu)體中除了startup函數(shù)指針,還有許多其它方法或字段。 其部分定義如下
struct _sapi_module_struct { char *name; // 名字(標(biāo)識(shí)用) char *pretty_name; // 更好理解的名字(自己翻譯的) int (*startup)(struct _sapi_module_struct *sapi_module); // 啟動(dòng)函數(shù) int (*shutdown)(struct _sapi_module_struct *sapi_module); // 關(guān)閉方法 int (*activate)(TSRMLS_D); // 激活 int (*deactivate)(TSRMLS_D); // 停用 int (*ub_write)(const char *str, unsigned int str_length TSRMLS_DC); // 不緩存的寫操作(unbuffered write) void (*flush)(void *server_context); // flush struct stat *(*get_stat)(TSRMLS_D); // get uid char *(*getenv)(char *name, size_t name_len TSRMLS_DC); // getenv void (*sapi_error)(int type, const char *error_msg, ...); /* error handler */ int (*header_handler)(sapi_header_struct *sapi_header, sapi_header_op_enum op,sapi_headers_struct *sapi_headers TSRMLS_DC); /* header handler */ /* send headers handler */ int (*send_headers)(sapi_headers_struct *sapi_headers TSRMLS_DC); void (*send_header)(sapi_header_struct *sapi_header, void *server_context TSRMLS_DC); /* send header handler */ int (*read_post)(char *buffer, uint count_bytes TSRMLS_DC); /* read POST data */ char *(*read_cookies)(TSRMLS_D); /* read Cookies */ /* register server variables */ void (*register_server_variables)(zval *track_vars_array TSRMLS_DC); void (*log_message)(char *message); /* Log message */ time_t (*get_request_time)(TSRMLS_D); /* Request Time */ void (*terminate_process)(TSRMLS_D); /* Child Terminate */ char *php_ini_path_override; // 覆蓋的ini路徑 ... ...};
以上的這些結(jié)構(gòu)在各服務(wù)器的接口實(shí)現(xiàn)中都有定義。如Apache2的定義:
static sapi_module_struct apache2_sapi_module = { 'apache2handler', 'Apache 2.0 Handler', php_apache2_startup,/* startup */ php_module_shutdown_wrapper, /* shutdown */ ...}
目前PHP內(nèi)置的很多SAPI實(shí)現(xiàn)都已不再維護(hù)或者變的有些非主流了,PHP社區(qū)目前正在考慮將一些SAPI移出代碼庫(kù)。 社區(qū)對(duì)很多功能的考慮是除非真的非常必要,或者某些功能已近非常通用了,否則就在PECL庫(kù)中, 例如非常流行的APC緩存擴(kuò)展將進(jìn)入核心代碼庫(kù)中。
整個(gè)SAPI類似于一個(gè)面向?qū)ο笾械哪0宸椒J降膽?yīng)用。 SAPI.c和SAPI.h文件所包含的一些函數(shù)就是模板方法模式中的抽象模板, 各個(gè)服務(wù)器對(duì)于sapi_module的定義及相關(guān)實(shí)現(xiàn)則是一個(gè)個(gè)具體的模板。
這樣的結(jié)構(gòu)在PHP的源碼中有多處使用, 比如在PHP擴(kuò)展開(kāi)發(fā)中,每個(gè)擴(kuò)展都需要定義一個(gè)zend_module_entry結(jié)構(gòu)體。 這個(gè)結(jié)構(gòu)體的作用與sapi_module_struct結(jié)構(gòu)體類似,都是一個(gè)類似模板方法模式的應(yīng)用。 在PHP的生命周期中如果需要調(diào)用某個(gè)擴(kuò)展,其調(diào)用的方法都是zend_module_entry結(jié)構(gòu)體中指定的方法, 如在上一小節(jié)中提到的在執(zhí)行各個(gè)擴(kuò)展的請(qǐng)求初始化時(shí),都是統(tǒng)一調(diào)用request_startup_func方法, 而在每個(gè)擴(kuò)展的定義時(shí),都通過(guò)宏P(guān)HP_RINIT指定request_startup_func對(duì)應(yīng)的函數(shù)。 以VLD擴(kuò)展為例:其請(qǐng)求初始化為PHP_RINIT(vld),與之對(duì)應(yīng)在擴(kuò)展中需要有這個(gè)函數(shù)的實(shí)現(xiàn):
PHP_RINIT_FUNCTION(vld) {}
所以, 我們?cè)趯憯U(kuò)展時(shí)也需要實(shí)現(xiàn)擴(kuò)展的這些接口,同樣,當(dāng)實(shí)現(xiàn)各服務(wù)器接口時(shí)也需要實(shí)現(xiàn)其對(duì)應(yīng)的SAPI。
相關(guān)文章:
1. python爬蟲(chóng)beautifulsoup解析html方法2. Python 如何將integer轉(zhuǎn)化為羅馬數(shù)(3999以內(nèi))3. python 實(shí)現(xiàn)aes256加密4. 詳解Python模塊化編程與裝飾器5. css進(jìn)階學(xué)習(xí) 選擇符6. Python性能測(cè)試工具Locust安裝及使用7. 以PHP代碼為實(shí)例詳解RabbitMQ消息隊(duì)列中間件的6種模式8. 使用Python解析Chrome瀏覽器書(shū)簽的示例9. html小技巧之td,div標(biāo)簽里內(nèi)容不換行10. python web框架的總結(jié)
