我個(gè)人認(rèn)為,Output buffering是比較純粹的4.0特征。盡管從概念上看來相當(dāng)簡(jiǎn)單,但是output buffering功能非常強(qiáng)大,能使開發(fā)者更容易地開發(fā)高級(jí)而有效的程序。
本文將介紹HTTP header,以及output buffering如何幫助您處理HTTP header,并介紹了output buffering的一些高級(jí)用法。
HTTP Header
對(duì) 于使用HTTP 協(xié)議建立的每個(gè)請(qǐng)求,Web服務(wù)器產(chǎn)生的響應(yīng)通常包括兩個(gè)部分 – 標(biāo)題和主體。例如,如果在Web服務(wù)器的文檔根目錄下有一個(gè)小文本文件,叫做example.txt,文件中包含文本Hello, world!,那么對(duì)此文件的HTTP 請(qǐng)求響應(yīng)如下所示:
代碼如下:
HTTP/1.1 200 OK
Date: Sat, 02 Sep 2000 21:40:08 GMT
Server: Apache/1.3.11 (Unix) mod_macro/1.1.1 PHP/4.0.2-dev
Last-Modified: Sat, 02 Sep 2000 21:39:49 GMT
ETag: "12600b-e-39b173a5"
Accept-Ranges: bytes
Content-Length: 14
Connection: close
Content-Type: text/plain
Hello, world!
這 個(gè)請(qǐng)求中的第一部分(就是較多的那部分)就是HTTP header。雖然用戶在瀏覽器中看不到HTTP header,但它包含了用于瀏覽器的信息,例如文檔內(nèi)容類型,使用的協(xié)議版本,文檔的最后更改日期等等。HTTP header并沒有太多的規(guī)則,通常情況下,它的格式如下:
代碼如下:
Field: Value[字段:值]
必須用空行將它們和文檔主體分開。
可以從PHP腳本添加或更改此HTTP header的信息。例如,可以使用 header() 函數(shù):
代碼如下:
header("Location: http://www.php.net/"); // 重定向到 http://www.php.net/
也可以使用 SetCookie() 函數(shù):
代碼如下:
SetCookie("foo", "bar");
你可能會(huì)知道HTTP cookies是使用 HTTP headers 來實(shí)現(xiàn)的。例如,以下PHP文件的 HTTP 請(qǐng)求響應(yīng)
代碼如下:
<?php
SetCookie("foo", "bar");
print "Set cookie.";
?>
將會(huì)是這樣的:
代碼如下:
HTTP/1.1 200 OK
Date: Sat, 02 Sep 2000 21:43:02 GMT
Server: Apache/1.3.11 (Unix) mod_macro/1.1.1PHP/4.0.2-dev
X-Powered-By: PHP/4.0.2-dev
Set-Cookie: foo=bar
Connection: close
Content-Type: text/html
Set cookie.
瀏覽器讀取從服務(wù)器返回的 HTTP header,知道送來了一個(gè)叫做 foo 的 cookie (在這里是一個(gè) session cookie),它的值是 bar。
為什么要使用Output Buffering技術(shù)
早 在PHP/FI 2.0時(shí)就很明顯需要output buffering技術(shù)了。如果你使用過這種版本的PHP,那么可能還記得經(jīng)常會(huì)碰到 Oops, SetCookie called after header has been sent 這個(gè)錯(cuò)誤消息,并使你捎頭抓耳,也弄不清是什么原因。
如 果你已使用過PHP的最新版本 -- PHP 3.0 甚至 PHP 4.0 -- 那么你會(huì)知道這個(gè)錯(cuò)誤消息: Oops, php_set_cookie called after header has been sent。或者,你在試圖調(diào)用 PHP 的 header() 函數(shù)時(shí)會(huì)遇到 Cannot add header information - headers already sent 消息。一般來說,output buffering技術(shù)用戶避免這些煩人的錯(cuò)誤消息,同時(shí)開發(fā)人員也可用于高級(jí)的用途。
這些錯(cuò)誤是什么時(shí)候產(chǎn)生的呢?如果你在已經(jīng)發(fā)送了HTTP header之后試圖添加或修改標(biāo)題信息,以及在文檔主體和標(biāo)題之間缺少空行時(shí),就會(huì)產(chǎn)生這些錯(cuò)誤消息。為了理解這是如何產(chǎn)生的,讓我們來看看PHP是如何處理HTTP header輸出和主體輸出的。
腳本開始執(zhí)行時(shí),它可以同時(shí)發(fā)送header(標(biāo)題)信息和主體信息。
Header信息(來自 header() 或 SetCookie() 函數(shù))并不會(huì)立即發(fā)送,相反,它被保存到一個(gè)列表中。
這樣就可以允許你修改標(biāo)題信息,包括缺省的標(biāo)題(例如 Content-Type 標(biāo)題)。但是,一旦腳本發(fā)送了任何非標(biāo)題的輸出(例如,使用塊或 print() 調(diào)用),那么PHP就必須先發(fā)送所有的標(biāo)題,然后再送出空行,終止 HTTP header,而在此之后才會(huì)繼續(xù)發(fā)送主體數(shù)據(jù)。從這時(shí)開始,任何添加或修改標(biāo)題信息的試圖都是不允許的,并會(huì)發(fā)送上述的錯(cuò)誤消息之一。
雖然這并不會(huì)引起多大的問題,有時(shí)候只是在發(fā)出任何輸入之前終止HTTP header,從而引起腳本邏輯的復(fù)雜化而已。Output buffering技術(shù)可以解決這些問題。
Output Buffering的工作原理
啟用output buffering時(shí),在腳本發(fā)送輸出時(shí),PHP并 不發(fā)送HTTP header。相反,它將此輸出通過管道(pipe)輸入到動(dòng)態(tài)增加的緩存中(只能在PHP 4.0中使用,它具有中央化的輸出機(jī)制)。你仍然可以修改,添加標(biāo)題行,或者設(shè)置cookie,因?yàn)闃?biāo)題實(shí)際上并沒有發(fā)送。最簡(jiǎn)單的情況是,當(dāng)腳本終止 時(shí),PHP將自動(dòng)發(fā)送HTTP header到瀏覽器,然后再發(fā)送輸出緩沖中的內(nèi)容。這簡(jiǎn)單吧。
基本用法
可以使用下面的四個(gè)函數(shù),它們可以幫助你控制output buffering:
代碼如下:
ob_start()
啟用output buffering機(jī)制。
Output buffering支持多層次 -- 例如,可以多次調(diào)用 ob_start() 函數(shù)。
ob_end_flush()
發(fā)送output buffer(輸出緩沖)并禁用output buffering機(jī)制。
ob_end_clean()
清除output buffer但不發(fā)送,并禁用output buffering。
ob_get_contents()
將當(dāng)前的output buffer返回成一個(gè)字符串。允許你處理腳本發(fā)出的任何輸出。
此外,可以啟用 php.ini 中的 output_buffering 指令。如果啟用了此指令,那么每個(gè)PHP腳本都相當(dāng)于一開始就調(diào)用了ob_start()函數(shù)。
Example 1
代碼如下:
<?php ob_start(); ?>
<h1>Example 1</h1>
<?php
print "Hello, $user ";
SetCookie("Wow", "This cookie has been set even though we've already emitted output!");
?>
這里,盡管你已發(fā)送了輸出(HTML代 碼塊中和 print 語句中),也可以使用 SetCookie() 調(diào)用,而不會(huì)出錯(cuò),真的要感謝output buffering機(jī)制。請(qǐng)注意使用output buffering機(jī)制用于這種目的會(huì)引起一定程度上的性能損失,因此最好缺省情況下不要啟用此機(jī)制。但是,對(duì)于復(fù)雜一些的腳本,output buffering可以簡(jiǎn)化邏輯性。
Example 2
代碼如下:
<?php
ob_start();
print "Here's a pretty dumb way to calculate the length of a string.";
$length = strlen(ob_get_buffer());
ob_end_clean();
?>
這個(gè)例子顯示了一個(gè)效率很低的確定字符串長(zhǎng)度的。它不是簡(jiǎn)單的使用strlen()函數(shù)處理,而是先啟用 output buffering 機(jī)制,將字符串打印出來,然后再確定output buffer的長(zhǎng)度。最后清除output buffer(并沒有發(fā)送),然后禁用output buffering機(jī)制。
更多信息請(qǐng)查看IT技術(shù)專欄