1. 我們從未手動(dòng)開(kāi)啟過(guò)PHP的相關(guān)進(jìn)程,它是隨著Apache的啟動(dòng)而運(yùn)行的;2. PHP通過(guò)mod_php5.so模塊和Apache相連(具體說(shuō)來(lái)是SAPI,即服務(wù)器應(yīng)用程序編程接口);3. PHP總共有三個(gè)模塊:內(nèi)核、Zend引擎、以及擴(kuò)展層;4. PHP內(nèi)核用來(lái)處理請(qǐng)求、文件流、錯(cuò)誤處理等相關(guān)操作;5. Zend引擎(ZE)用以將源文件轉(zhuǎn)換成機(jī)器語(yǔ)言,然后在虛擬機(jī)上運(yùn)行它;6. 擴(kuò)展層是一組函數(shù)、類(lèi)庫(kù)和流,PHP使用它們來(lái)執(zhí)行一些特定的操作。比如,我們需要mysql擴(kuò)展來(lái)連接MySQL數(shù)據(jù)庫(kù);7. 當(dāng)ZE執(zhí)行程序時(shí)可能會(huì)需要連接若干擴(kuò)展,這時(shí)ZE將控制權(quán)交給擴(kuò)展,等處理完特定任務(wù)后再返還;8. 最后,ZE將程序運(yùn)行結(jié)果返回給PHP內(nèi)核,它再將結(jié)果傳送給SAPI層,最終輸出到瀏覽器上。
創(chuàng)新互聯(lián)建站主營(yíng)盤(pán)龍網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營(yíng)網(wǎng)站建設(shè)方案,成都APP應(yīng)用開(kāi)發(fā),盤(pán)龍h5成都小程序開(kāi)發(fā)搭建,盤(pán)龍網(wǎng)站營(yíng)銷(xiāo)推廣歡迎盤(pán)龍等地區(qū)企業(yè)咨詢(xún)
深入探討
等等,沒(méi)有這么簡(jiǎn)單。以上過(guò)程只是個(gè)簡(jiǎn)略版,讓我們?cè)偕钊胪诰蛞幌?,看看幕后還發(fā)生了些什么。
1. Apache啟動(dòng)后,PHP解釋程序也隨之啟動(dòng);
2. PHP的啟動(dòng)過(guò)程有兩步;
3. 第一步是初始化一些環(huán)境變量,這將在整個(gè)SAPI生命周期中發(fā)生作用;4. 第二步是生成只針對(duì)當(dāng)前請(qǐng)求的一些變量設(shè)置。
PHP啟動(dòng)第一步
不清楚什么第一第二步是什么?別擔(dān)心,我們接下來(lái)詳細(xì)討論一下。讓我們先看看第一步,也是最主要的一步。要記住的是,第一步的操作在任何請(qǐng)求到達(dá)之前就發(fā)生了。
1. 啟動(dòng)Apache后,PHP解釋程序也隨之啟動(dòng);
2. PHP調(diào)用各個(gè)擴(kuò)展的MINIT方法,從而使這些擴(kuò)展切換到可用狀態(tài)。看看php.ini文件里打開(kāi)了哪些擴(kuò)展吧;3. MINIT的意思是“模塊初始化”。各個(gè)模塊都定義了一組函數(shù)、類(lèi)庫(kù)等用以處理其他請(qǐng)求。
一個(gè)典型的MINIT方法如下:
PHP_MINIT_FUNCTION(extension_name){
}
PHP啟動(dòng)第二步
1. 當(dāng)一個(gè)頁(yè)面請(qǐng)求發(fā)生時(shí),SAPI層將控制權(quán)交給PHP層。于是PHP設(shè)置了用于回復(fù)本次請(qǐng)求所需的環(huán)境變量。同時(shí),它還建立一個(gè)變量表,用來(lái)存放執(zhí)行過(guò)程中產(chǎn)生的變量名和值。
2. PHP調(diào)用各個(gè)模塊的RINIT方法,即“請(qǐng)求初始化”。一個(gè)經(jīng)典的例子是Session模塊的RINIT,如果在php.ini中啟用了Session模塊,那在調(diào)用該模塊的RINIT時(shí)就會(huì)初始化$_SESSION變量,并將相關(guān)內(nèi)容讀入;3. RINIT方法可以看作是一個(gè)準(zhǔn)備過(guò)程,在程序執(zhí)行之間就會(huì)自動(dòng)啟動(dòng)。
一個(gè)典型的RINIT方法如下:
PHP_RINIT_FUNCTION(extension_name) {
}
PHP關(guān)閉第一步
如同PHP啟動(dòng)一樣,PHP的關(guān)閉也分兩步:
1. 一旦頁(yè)面執(zhí)行完畢(無(wú)論是執(zhí)行到了文件末尾還是用exit或die函數(shù)中止),PHP就會(huì)啟動(dòng)清理程序。它會(huì)按順序調(diào)用各個(gè)模塊的RSHUTDOWN方法。
2. RSHUTDOWN用以清除程序運(yùn)行時(shí)產(chǎn)生的符號(hào)表,也就是對(duì)每個(gè)變量調(diào)用unset函數(shù)。
一個(gè)典型的RSHUTDOWN方法如下:
PHP_RSHUTDOWN_FUNCTION(extension_name) {
}
PHP關(guān)閉第二步
最后,所有的請(qǐng)求都已處理完畢,SAPI也準(zhǔn)備關(guān)閉了,PHP開(kāi)始執(zhí)行第二步:
1. PHP調(diào)用每個(gè)擴(kuò)展的MSHUTDOWN方法,這是各個(gè)模塊最后一次釋放內(nèi)存的機(jī)會(huì)。
一個(gè)典型的RSHUTDOWN方法如下:
PHP_MSHUTDOWN_FUNCTION(extension_name) {
}
在現(xiàn)代 PHP 特性中,流或許是最出色但使用率最低的。雖然 PHP 4.3 就引入了流,但是很多開(kāi)發(fā)者并不知道流的存在,因?yàn)槿藗兒苌偬峒傲鳎伊鞯奈臋n也很匱乏。PHP 官方文檔對(duì)流的解釋如下:
可能看完這段解釋后還是云里霧里,我們簡(jiǎn)化一下,流的作用是在出發(fā)地和目的地之間傳輸數(shù)據(jù)。出發(fā)地和目的地可以是文件、命令行進(jìn)程、網(wǎng)絡(luò)連接、ZIP 或 TAR 壓縮文件、臨時(shí)內(nèi)存、標(biāo)準(zhǔn)輸入或輸出,或者是通過(guò) PHP 流封裝協(xié)議實(shí)現(xiàn)的任何其他資源。
如果你讀寫(xiě)過(guò)文件,就用過(guò)流;如果你從 php://stdin 讀取過(guò)數(shù)據(jù),或者把輸入寫(xiě)入 php://stdout ,也用過(guò)流。流為 PHP 的很多 IO 函數(shù)提供了底層實(shí)現(xiàn),如 file_get_contents、fopn、fread 和 fwrite 等。PHP 的流函數(shù)提供了不同資源的統(tǒng)一接口。
我們可以把流比作管道,把水(資源數(shù)據(jù))從一個(gè)地方引到另一個(gè)地方。在水從出發(fā)地到目的地的過(guò)程中,我們可以過(guò)濾水,可以改變水質(zhì),可以添加水,也可以排出水。
流式數(shù)據(jù)的種類(lèi)各異,每種類(lèi)型需要獨(dú)特的協(xié)議,以便讀寫(xiě)數(shù)據(jù),我們稱(chēng)這些協(xié)議為 流封裝協(xié)議 。例如,我們可以讀寫(xiě)文件系統(tǒng),可以通過(guò) HTTP、HTTPS 或 SSH 與遠(yuǎn)程 Web 服務(wù)器通信,還可以打開(kāi)并讀寫(xiě) ZIP、RAR 或 PHAR 壓縮文件。這些通信方式都包含下述相同的過(guò)程:
1.開(kāi)始通信
2.讀取數(shù)據(jù)
3.寫(xiě)入數(shù)據(jù)
4.結(jié)束通信
雖然過(guò)程是一樣的,但是讀寫(xiě)文件系統(tǒng)中文件的方式與收發(fā) HTTP 消息的方式有所不同,流封裝協(xié)議的作用是使用通用的接口封裝這種差異。
每個(gè)流都有一個(gè)協(xié)議和一個(gè)目標(biāo)。指定協(xié)議和目標(biāo)的方法是使用流標(biāo)識(shí)符:scheme://target,其中 scheme 是流的封裝協(xié)議,target 是流的數(shù)據(jù)源。
http://流封裝協(xié)議
下面使用 HTTP 流封裝協(xié)議創(chuàng)建了一個(gè)與 Flicker API 通信的 PHP 流:
不要以為這是普通的網(wǎng)頁(yè) URL,file_get_contents() 函數(shù)的字符串參數(shù)其實(shí)是一個(gè)流標(biāo)識(shí)符。http 協(xié)議會(huì)讓 PHP 使用 HTTP 流封裝協(xié)議,在這個(gè)參數(shù)中,http 之后是流的目標(biāo)。
我們通常使用 file_get_contents()、fopen()、fwrite() 和 fclose() 等函數(shù)讀寫(xiě)文件系統(tǒng),因?yàn)?PHP 默認(rèn)使用的流封裝協(xié)議是 file://,所以我們很少認(rèn)為這些函數(shù)使用的是 PHP 流。下面的示例演示了使用 file:// 流封裝協(xié)議創(chuàng)建一個(gè)讀寫(xiě) /etc/hosts 文件的流:
我們通常會(huì)省略掉 file:// 協(xié)議,因?yàn)檫@是 PHP 使用的默認(rèn)值。
php://流封裝協(xié)議
編寫(xiě)命令行腳本的 PHP 開(kāi)發(fā)者會(huì)感激 php:// 流封裝協(xié)議,這個(gè)流封裝協(xié)議的作用是與 PHP 腳本的標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤文件描述符通信。我們可以使用 PHP 提供的文件系統(tǒng)函數(shù)打開(kāi)、讀取或?qū)懭胂旅嫠膫€(gè)流:
1. php://stdin :這是個(gè)只讀 PHP 流,其中的數(shù)據(jù)來(lái)自標(biāo)準(zhǔn)輸入。PHP 腳本可以使用這個(gè)流接收命令行傳入腳本的信息;
2. php://stdout :把數(shù)據(jù)寫(xiě)入當(dāng)前的輸出緩沖區(qū),這個(gè)流只能寫(xiě),無(wú)法讀或?qū)ぶ罚?/p>
3. php://memory :從系統(tǒng)內(nèi)存中讀取數(shù)據(jù),或者把數(shù)據(jù)寫(xiě)入系統(tǒng)內(nèi)存。缺點(diǎn)是系統(tǒng)內(nèi)存有限,所有使用 php://temp 更安全;
4. php://temp :和 php://memory 類(lèi)似,不過(guò),沒(méi)有可用內(nèi)存時(shí),PHP 會(huì)把數(shù)據(jù)寫(xiě)入這個(gè)臨時(shí)文件。
其他流封裝協(xié)議
PHP 和 PHP 擴(kuò)展還提供了很多其他流封裝協(xié)議,例如,與 ZIP 和 TAR 壓縮文件、FTP 服務(wù)器、數(shù)據(jù)壓縮庫(kù)、Amazon API、Dropbox API 等通信的流封裝協(xié)議。需要注意的是,PHP 中的 fopen()、fgets()、fputs()、feof() 以及 fclose() 等函數(shù)不僅可以用來(lái)處理文件系統(tǒng)中的文件,還可以在所有支持這些函數(shù)的流封裝協(xié)議中使用。
自定義流封裝協(xié)議
我們還可以自己編寫(xiě) PHP 流封裝協(xié)議。PHP 提供了一個(gè)示例 StreamWrapper 類(lèi),演示如何編寫(xiě)自定義的流封裝協(xié)議,支持部分或全部 PHP 文件系統(tǒng)函數(shù)。關(guān)于如何編寫(xiě),具體請(qǐng)參考以下文檔:
有些 PHP 流能夠接受一系列可選的參數(shù),這些參數(shù)叫流上下文,用于定制流的行為。不同的流封裝協(xié)議使用的流上下文有所不同,流上下文使用 stream_context_create() 函數(shù)創(chuàng)建,這個(gè)函數(shù)返回的上下文對(duì)象可以傳入大多數(shù)文件系統(tǒng)函數(shù)。
例如,你知道可以使用 file_get_contents() 發(fā)送 HTTP POST 請(qǐng)求嗎?使用一個(gè)流上下文對(duì)象即可實(shí)現(xiàn):
流過(guò)濾器
目前為止我們討論了如何打開(kāi)流,讀取流中的數(shù)據(jù),以及把數(shù)據(jù)寫(xiě)入流。不過(guò),PHP 流真正強(qiáng)大的地方在于過(guò)濾、轉(zhuǎn)換、添加或刪除流中傳輸?shù)臄?shù)據(jù),例如,我們可以打開(kāi)一個(gè)流處理 Markdown 文件,在把文件內(nèi)容讀入內(nèi)存的過(guò)程中自動(dòng)將其轉(zhuǎn)化為 HTML。
運(yùn)行該腳本,輸出的都是大寫(xiě)字母:
我們還可以使用 php://filter 流封裝協(xié)議把過(guò)濾器附加到流上,不過(guò),使用這種方式之前必須先打開(kāi) PHP 流:
這個(gè)方式實(shí)現(xiàn)效果和 stream_filter_append() 函數(shù)一樣,但是相比之下更為繁瑣。不過(guò),PHP 的某些文件系統(tǒng)函數(shù)在調(diào)用后無(wú)法附加過(guò)濾器,例如 file() 和 fpassthru(),使用這些函數(shù)時(shí)只能使用 php://filter 流封裝協(xié)議附加流過(guò)濾器。
自定義流過(guò)濾器
我們還可以編寫(xiě)自定義的流過(guò)濾器。其實(shí),大多數(shù)情況下都要使用自定義的流過(guò)濾器,自定義的流過(guò)濾器是個(gè) PHP 類(lèi),繼承內(nèi)置的 php_user_filter 類(lèi)( ),且必須實(shí)現(xiàn) filter()、onCreate() 和 onClose() 方法,最后,必須使用 stream_filter_register() 函數(shù)注冊(cè)自定義的流過(guò)濾器。
然后,我們必須使用 stream_filter_register() 函數(shù)注冊(cè)這個(gè)自定義的 DirtyWordsFilter 流過(guò)濾器:
第一個(gè)參數(shù)用于標(biāo)識(shí)這個(gè)自定義過(guò)濾器的過(guò)濾器名,第二個(gè)參數(shù)是這個(gè)自定義過(guò)濾器的類(lèi)名。接下來(lái)就可以使用這個(gè)自定義的流過(guò)濾器了:
修改 test.txt 內(nèi)容如下:
運(yùn)行上面的自定義過(guò)濾器腳本,結(jié)果如下:
stream_bucket_append函數(shù):為隊(duì)列添加數(shù)據(jù)
stream_bucket_make_writeable函數(shù):從操作的隊(duì)列中返回一個(gè)數(shù)據(jù)對(duì)象
stream_bucket_new函數(shù):為當(dāng)前隊(duì)列創(chuàng)建一個(gè)新的數(shù)據(jù)
stream_bucket_prepend函數(shù):預(yù)備數(shù)據(jù)到隊(duì)列
stream_context_create函數(shù):創(chuàng)建數(shù)據(jù)流上下文
stream_context_get_default函數(shù):獲取默認(rèn)的數(shù)據(jù)流上下文
stream_context_get_options函數(shù):獲取數(shù)據(jù)流的設(shè)置
stream_context_set_option函數(shù):對(duì)數(shù)據(jù)流、數(shù)據(jù)包或者上下文進(jìn)行設(shè)置
stream_context_set_params函數(shù):為數(shù)據(jù)流、數(shù)據(jù)包或者上下文設(shè)置參數(shù)
stream_copy_to_stream函數(shù):在數(shù)據(jù)流之間進(jìn)行復(fù)制操作
stream_filter_append函數(shù):為數(shù)據(jù)流添加過(guò)濾器
stream_filter_prepend函數(shù):為數(shù)據(jù)流預(yù)備添加過(guò)濾器
stream_filter_register函數(shù):注冊(cè)一個(gè)數(shù)據(jù)流的過(guò)濾器并作為PHP類(lèi)執(zhí)行
stream_filter_remove函數(shù):從一個(gè)數(shù)據(jù)流中移除過(guò)濾器
stream_get_contents函數(shù):讀取數(shù)據(jù)流中的剩余數(shù)據(jù)到字符串
stream_get_filters函數(shù):返回已經(jīng)注冊(cè)的數(shù)據(jù)流過(guò)濾器列表
stream_get_line函數(shù):按照給定的定界符從數(shù)據(jù)流資源中獲取行
stream_get_meta_data函數(shù):從封裝協(xié)議文件指針中獲取報(bào)頭/元數(shù)據(jù)
stream_get_transports函數(shù):返回注冊(cè)的Socket傳輸列表
stream_get_wrappers函數(shù):返回注冊(cè)的數(shù)據(jù)流列表
stream_register_wrapper函數(shù):注冊(cè)一個(gè)用PHP類(lèi)實(shí)現(xiàn)的URL封裝協(xié)議
stream_select函數(shù):接收數(shù)據(jù)流數(shù)組并等待它們狀態(tài)的改變
stream_set_blocking函數(shù):將一個(gè)數(shù)據(jù)流設(shè)置為堵塞或者非堵塞狀態(tài)
stream_set_timeout函數(shù):對(duì)數(shù)據(jù)流進(jìn)行超時(shí)設(shè)置
stream_set_write_buffer函數(shù):為數(shù)據(jù)流設(shè)置緩沖區(qū)
stream_socket_accept函數(shù):接受由函數(shù)stream_ socket_server()創(chuàng)建的Socket連接
stream_socket_client函數(shù):打開(kāi)網(wǎng)絡(luò)或者UNIX主機(jī)的Socket連接
stream_socket_enable_crypto函數(shù):為一個(gè)已經(jīng)連接的Socket打開(kāi)或者關(guān)閉數(shù)據(jù)加密
stream_socket_get_name函數(shù):獲取本地或者網(wǎng)絡(luò)Socket的名稱(chēng)
stream_socket_pair函數(shù):創(chuàng)建兩個(gè)無(wú)區(qū)別的Socket數(shù)據(jù)流連接
stream_socket_recvfrom函數(shù):從Socket獲取數(shù)據(jù),不管其連接與否
stream_socket_sendto函數(shù):向Socket發(fā)送數(shù)據(jù),不管其連接與否
stream_socket_server函數(shù):創(chuàng)建一個(gè)網(wǎng)絡(luò)或者UNIX Socket服務(wù)端
stream_wrapper_restore函數(shù):恢復(fù)一個(gè)事先注銷(xiāo)的數(shù)據(jù)包
stream_wrapper_unregister函數(shù):注銷(xiāo)一個(gè)URL地址包
整合資料
本文整合于以下兩篇文章
php://stdin, php://stdout 和 php://stderrphp://stdin,php://stdout 和 php://stderr允許訪問(wèn) PHP 進(jìn)程相應(yīng)的輸入或者輸出流。php://inputphp://input 是個(gè)可以訪問(wèn)請(qǐng)求的原始數(shù)據(jù)的只讀流。 POST 請(qǐng)求的情況下,最好使用 php://input 來(lái)代替 $HTTP_RAW_POST_DATA(原生的post數(shù)據(jù)),因?yàn)樗灰蕾?lài)于特定的 php.ini 指令,內(nèi)存消耗更少。如下例:結(jié)果:php://outputphp://output 是一個(gè)只寫(xiě)的數(shù)據(jù)流, 允許你以 print 和 echo 一樣的方式 寫(xiě)入到輸出緩沖區(qū)。php://fdphp://fd 允許直接訪問(wèn)指定的文件描述符。 例如 php://fd/3 引用了文件描述符 3。php://memory 和 php://tempphp://memory 和 php://temp 是一個(gè)類(lèi)似文件 包裝器的數(shù)據(jù)流,允許讀寫(xiě)臨時(shí)數(shù)據(jù)。 兩者的唯一區(qū)別是 php://memory 總是把數(shù)據(jù)儲(chǔ)存在內(nèi)存中, 而 php://temp 會(huì)在內(nèi)存量達(dá)到預(yù)定義的限制后(默認(rèn)是 2MB)存入臨時(shí)文件中。 臨時(shí)文件位置的決定和 sys_get_temp_dir() 的方式一致。php://filterphp://filter 是一種元封裝器, 設(shè)計(jì)用于數(shù)據(jù)流打開(kāi)時(shí)的篩選過(guò)濾應(yīng)用。 這對(duì)于一體式(all-in-one)的文件函數(shù)非常有用,類(lèi)似 readfile()、 file() 和 file_get_contents(), 在數(shù)據(jù)流內(nèi)容讀取之前沒(méi)有機(jī)會(huì)應(yīng)用其他過(guò)濾器。參數(shù)如下:如下例:
您好,PHP輸出數(shù)據(jù)有四種方式,echo、var_dump、print_r、retrun(這個(gè)一般是PHP框架常用)