PHP的啟動與終止


1.2 PHP的啟動與終止

PHP程序的啟動可以看作有兩個概念上的啟動,終止也有兩個概念上的終止。 其中一個是PHP作為Apache(拿它舉例,板磚勿扔)的一個模塊的啟動與終止, 這次啟動php會初始化一些必要數據,比如與宿主Apache有關的,並且這些數據是常駐內存的! 終止與之相對。 還有一個概念上的啟動就是當Apache分配一個頁面請求過來的時候,PHP會有一次啟動與終止,這也是我們最常討論的一種。

現在我們主要來看一個PHP擴展的生命旅程是怎樣走完這四個過程的。

在最初的初始化時候,就是PHP隨着Apache的啟動而誕生在內存里的時候, 它會把自己所有已加載擴展的MINIT方法(全稱Module Initialization,是由每個模塊自己定義的函數。)都執行一遍。 在這個時間里,擴展可以定義一些自己的常量、類、資源等所有會被用戶端的PHP腳本用到的東西。 但你要記住,這里定義的東東都會隨着Apache常駐內存,可以被所有請求使用,直到Apache卸載掉PHP模塊!

內核中預置了PHP_MINIT_FUNCTION宏函數,來幫助我們實現這個功能:

?
1
2
3
4
5
6
7
8
//拋棄作者那個例子,書才看兩頁整那樣的例子太復雜了!
//walu是我擴展的名稱
int time_of_minit; //在MINIT()中初始化,在每次頁面請求中輸出,看看是否變化
PHP_MINIT_FUNCTION(walu)
{
     time_of_minit= time (NULL); //我們在MINIT啟動中對他初始化
     return SUCCESS; //返回SUCCESS代表正常,返回FALIURE就不會加載這個擴展了。
}

當一個頁面請求到來時候,PHP會迅速的開辟一個新的環境,並重新掃描自己的各個擴展, 遍歷執行它們各自的RINIT方法(俗稱Request Initialization), 這時候一個擴展可能會初始化在本次請求中會使用到的變量等, 還會初始化等會兒用戶端(即PHP腳本)中的變量之類的,內核預置了PHP_RINIT_FUNCTION()這個宏函數來幫我們實現這個功能:

?
1
2
3
4
5
6
int time_of_rinit; //在RINIT里初始化,看看每次頁面請求的時候變不。
PHP_RINIT_FUNCTION(walu)
{
     time_of_rinit= time (NULL);
     return SUCCESS;
}

好了,現在這個頁面請求執行的差不多了,可能是順利的走到了自己文件的最后, 也可能是出師未捷,半道被用戶給die或者exit了, 這時候PHP便會啟動回收程序,收拾這個請求留下的爛攤子。 它這次會執行所有已加載擴展的RSHUTDOWN(俗稱Request Shutdown)方法, 這時候擴展可以抓緊利用內核中的變量表之類的做一些事情, 因為一旦PHP把所有擴展的RSHUTDOWN方法執行完, 便會釋放掉這次請求使用過的所有東西, 包括變量表的所有變量、所有在這次請求中申請的內存等等。

內核預置了PHP_RSHUTDOWN_FUNCTION宏函數來幫助我們實現這個功能

?
1
2
3
4
5
6
7
PHP_RSHUTDOWN_FUNCTION(walu)
{
     FILE *fp= fopen ( "time_rshutdown.txt" , "a+" );
     fprintf (fp, "%ld\n" , time (NULL)); //讓我們看看是不是每次請求結束都會在這個文件里追加數據
     fclose (fp);
     return SUCCESS;
}

前面該啟動的也啟動了,該結束的也結束了,現在該Apache老人家歇歇的時候,當Apache通知PHP自己要Stop的時候,PHP便進入MSHUTDOWN(俗稱Module Shutdown)階段。這時候PHP便會給所有擴展下最后通牒,如果哪個擴展還有未了的心願,就放在自己MSHUTDOWN方法里,這可是最后的機會了,一旦PHP把擴展的MSHUTDOWN執行完,便會進入自毀程序,這里一定要把自己擅自申請的內存給釋放掉,否則就杯具了。

內核中預置了PHP_MSHUTDOWN_FUNCTION宏函數來幫助我們實現這個功能:

?
1
2
3
4
5
6
PHP_MSHUTDOWN_FUNCTION(walu)
{
     FILE *fp= fopen ( "time_mshutdown.txt" , "a+" );
     fprintf (fp, "%ld\n" , time (NULL));
     return SUCCESS;
}

這四個宏都是在walu.c里完成最終實現的,而他們的則是在/main/php.h里被定義的(其實也是調用的別的宏,本節最后我把這幾個宏給展開了,供有需要的人查看)。

好了,現在我們本節內容說完了,下面我們把所有的代碼合在一起,並預測一下應該出現的結果:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
//這些代碼都在walu.c里面,不再.h里
 
int time_of_minit; //在MINIT中初始化,在每次頁面請求中輸出,看看是否變化
PHP_MINIT_FUNCTION(walu)
{
     time_of_minit= time (NULL); //我們在MINIT啟動中對他初始化
     return SUCCESS;
}
 
int time_of_rinit; //在RINIT里初始化,看看每次頁面請求的時候變不。
PHP_RINIT_FUNCTION(walu)
{
     time_of_rinit= time (NULL);
     return SUCCESS;
}
 
PHP_RSHUTDOWN_FUNCTION(walu)
{
     FILE *fp= fopen ( "/cnan/www/erzha/time_rshutdown.txt" , "a+" ); //請確保文件可寫,否則apache會莫名崩潰
     fprintf (fp, "%d\n" , time (NULL)); //讓我們看看是不是每次請求結束都會在這個文件里追加數據
     fclose (fp);
     return SUCCESS;
}
 
PHP_MSHUTDOWN_FUNCTION(walu)
{
     FILE *fp= fopen ( "/cnan/www/erzha/time_mshutdown.txt" , "a+" ); //請確保文件可寫,否則apache會莫名崩潰
     fprintf (fp, "%d\n" , time (NULL));
     return SUCCESS;
}
 
//我們在頁面里輸出time_of_minit和time_of_rinit的值
PHP_FUNCTION(walu_test)
{
     php_printf( "%d<br />" ,time_of_minit);
     php_printf( "%d<br />" ,time_of_rinit);
     return ;
}
  • time_of_minit的值每次請求都不變。
  • time_of_rinit的值每次請求都改變。
  • 每次頁面請求都會往time_rshutdown.txt中寫入數據。
  • 只有在apache結束后time_mshutdown.txt才寫入有數據。

多謝 閘北陸小洪 指出的有關time_of_rinit的筆誤。

上面便是PHP中典型的啟動-終止模型,實際情況可能因為模式不同而有所變化, 到底PHP的啟動-終止會有多少中不同變化方式,請看下一節。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM