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的啟動-終止會有多少中不同變化方式,請看下一節。
