PHP的最多的兩種運行模式是WEB模式、CLI模式。
無論哪種模式,PHP工作原理都是一樣的,作為一種SAPI運行。
1、當我們在終端敲入php這個命令的時候,它使用的是CLI。
它就像一個web服務器一樣來支持php完成這個請求,請求完成後再重新把控制權交給終端。
2、當使用Apache作為宿主時,當一個請求到來時,PHP會來支持完成這個請求。
main/php.h中定義了以下幾個宏
#define PHP_MINIT_FUNCTION ZEND_MODULE_STARTUP_D
#define PHP_MSHUTDOWN_FUNCTION ZEND_MODULE_SHUTDOWN_D
#define PHP_RINIT_FUNCTION ZEND_MODULE_ACTIVATE_D
#define PHP_RSHUTDOWN_FUNCTION ZEND_MODULE_DEACTIVATE_D
#define PHP_MINFO_FUNCTION ZEND_MODULE_INFO_D
#define PHP_GINIT_FUNCTION ZEND_GINIT_FUNCTION
#define PHP_GSHUTDOWN_FUNCTION ZEND_GSHUTDOWN_FUNCTION
對應的作用是
PHP_MINIT_FUNCTION 初始化module時運行
PHP_MSHUTDOWN_FUNCTION 當module被卸載時運行
PHP_RINIT_FUNCTION 當一個REQUEST請求初始化時運行
PHP_RSHUTDOWN_FUNCTION 當一個REQUEST請求結束時運行
PHP_MINFO_FUNCTION 這個是設置phpinfo中這個模塊的信息
PHP_GINIT_FUNCTION 初始化全局變量時
PHP_GSHUTDOWN_FUNCTION 釋放全局變量時
看一個自定義擴展案例片段:
www.2cto.com
int minit_time;
PHP_MINIT_FUNCTION(test)
{
minit_time = time(NULL);
return SUCCESS;
}
PHP_MSHUTDOWN_FUNCTION(test)
{
FILE *fp=fopen("mshutdown.txt","a+");
fprintf(fp,"%ld\n",time(NULL));//讓我們看看是不是每次請求結束都會在這個文件裡追加數據
fclose(fp);
return SUCCESS;
}
int rinit_time;
PHP_RINIT_FUNCTION(test)
{
rinit_time = time(NULL);
return SUCCESS;
}
PHP_RSHUTDOWN_FUNCTION(test)
{
FILE *fp=fopen("rshutdown.txt","a+");
fprintf(fp,"%ld\n",time(NULL));//讓我們看看是不是每次請求結束都會在這個文件裡追加數據
fclose(fp);
return SUCCESS;
}
PHP_MINFO_FUNCTION(test)
{
php_info_print_table_start();//調用php_write輸出HTML標簽
php_info_print_table_header(2, "module info", "enabled");
php_info_print_table_end();//調用php_write輸出HTML標簽
/* Remove comments if you have entries in php.ini
DISPLAY_INI_ENTRIES();
*/
}
//定義PHP中可以調用的函數test(),讓它在頁面裡輸出minit_time和rinit_time的值
PHP_FUNCTION(test)
{
php_printf("%d<br />",time_of_minit);
php_printf("%d<br />",time_of_rinit);
return;
}
以Apache為例,
如果在多線程的模式下工作:
在這種模式下,只有一個服務器進程在運行著,但會同時運行很多線程,這樣可以減少一些資源開銷,向Module init和Module shutdown就只需要運行一遍就行了,一些全局變量也只需要初始化一次,因為線程獨具的特質,使得各個請求之間方便的共享一些數據成為可能。
多線程工作方式如下圖
:
如果在多進程的模式下工作:
minit_time、rinit_time的值每次請求都在變。
每次頁面請求結束都會往time_rshutdown.txt、time_mshutdown.txt中寫入數據。
下面是多進程工作模式圖: