PHP ob_start()與ob_gzhandler()
Output Control 函數可以讓你自由控制腳本中數據的輸出。它非常地有用,特別是對于:當你想在數據已經輸出后,再輸出文件頭的情況。輸出控制函數不對使用 header() 或 setcookie(),
發送的文件頭信息產生影響,只對那些類似于 echo() 和 PHP 代碼的數據塊有作用。
ob_start(); //打開緩沖區
echo "Hello\n"; //輸出
header(“location:index.php”); //把瀏覽器重定向到index.php
ob_end_flush();//輸出全部內容到瀏覽器
所有對header()函數有了解的人都知道,這個函數會發送一段文件頭給瀏覽器,但是如果在使用這個函數之前已經有了任何輸出(包括空輸出,比如空格,回車和換行)就會提示出錯。如果我們去掉第一行的ob_start(),再執行此程序,我們會發現得到了一條錯誤提示:“Header had all ready send by”!但是加上ob_start,就不會提示出錯,原因是當打開了緩沖區,echo后面的字符不會輸出到瀏覽器,而是保留在服務器,直到你使用flush或者ob_end_flush才會輸出,所以并不會有任何文件頭輸出的錯誤!
for($i = 1; $i <= 300; $i++ ) print(“ “);
// 這一句話非常關鍵,cache的結構使得它的內容只有達到一定的大小才能從瀏覽器里輸出 換言之,如果cache的內容不達到一定的大小,它是不會在程序執行完畢前輸出的。經 過測試,我發現這個大小的底限是256個字符長。這意味著cache以后接收的內容都會 源源不斷的被發送出去。
For($j = 1; $j <= 20; $j++) {
echo $j.””;
flush(); //這一部會使cache新增的內容被擠出去,顯示到瀏覽器上
sleep(1); //讓程序“睡”一秒鐘,會讓你把效果看得更清楚
}
?>
注:如果在程序的首部加入ob_implicit_flush()打開絕對刷新,就可以在程序中不再使用flush(),這樣做的好處是:提高效率!
2. 關于ob系列函數:
我想先引用我的好朋友y10k的一個例子:
比如你用得到服務器和客戶端的設置信息,但是這個信息會因為客戶端的不同而不同,如果想要保存phpinfo()函數的輸出怎么辦呢?在沒有緩沖區控制之前,可以說一點辦法也沒有,但是有了
緩沖區的控制,我們可以輕松的解決:
-------------------------------------------------------------
ob_start(); //打開緩沖區
phpinfo(); //使用phpinfo函數
$info=ob_get_contents(); //得到緩沖區的內容并且賦值給$info
$file=fopen('info.txt','w'); //打開文件info.txt
fwrite($file,$info); //寫入信息到info.txt
fclose($file); //關閉文件info.txt
?>
-------------------------------------------------------------
用以上的方法,就可以把不同用戶的phpinfo信息保存下來,這在以前恐怕沒有辦法辦到!其實上
面就是將一些“過程”轉化為“函數”的方法!
可能現在大家對ob_start()的功能有了一定的了解,上面的一個例子看似簡單,但實際上已經掌握了使用ob_start()的要點。
<1>.使用ob_start打開browser的cache,這樣可以保證cache的內容在你調用flush(),ob_end_flush()(或程序執行完畢)之前不會被輸出。
<2>.現在的你應該知道你所擁有的優勢:可以在任何輸出內容后面使用header,setcookie以及session,這是ob_start一個很大的特點;也可以使用ob_start的參數,在cache被寫入后,然后自動運行命令,比如ob_start("ob_gzhandler");而我們最常用的做法是用ob_get_contents()得
到cache中的內容,然后再進行處理……
<3>.當處理完畢后,我們可以使用各種方法輸出,flush(),ob_end_flush(),以及等到程序執行完畢后的自動輸出。當然,如果你用的是ob_get_contents(),那么就要你自己控制輸出方式了。
來,讓我們看看能用ob系列函數做些什么……
一、 靜態模版技術
簡介:所謂靜態模版技術就是通過某種方式,使得用戶在client端得到的是由PHP產生的html頁面。如果這個html頁面不會再被更新,那么當另外的用戶再次瀏覽此頁面時,程序將不會再調用PHP以及相關的數據庫,對于某些信息量比較大的網站,例如sina,163,sohu。類似這種的技術帶來的好處是非常巨大的。
我所知道的實現靜態輸出的有兩種辦法:
<1>.通過y10k修改的phplib的一個叫template.inc.php類實現。
<2>.使用ob系列函數實現。
對于第一種方法,因為不是這篇文章所要研究的問題,所以不再贅述。
我們現在來看一看第二種方法的具體實現:
ob_start();//打開緩沖區
php頁面的全部輸出
$content = ob_get_contents();//取得php頁面輸出的全部內容
$fp = fopen(“output00001.html”, “w”); //創建一個文件,并打開,準備寫入 fwrite($fp, $content); //把php頁面的內容全部寫入output00001.html,然后……
fclose($fp);
PHP4.0提供了一個輸出緩沖函數集合。輸出緩沖支持允許你寫包裹函數功能壓縮緩沖區。在PHP4的輸出緩沖支持允許HTML頭信息存放,無論 HTML的正文是否輸出。但在PHP中,頭信息header(), content type, and cookies不采用緩沖。
在使用PHP的過程中不免要使用到header和setcookie兩個函數,這兩個函數會發送一段文件頭信息給瀏覽器,但是如果在使用這兩個函數之前已經有了任何輸出(包括空輸出,比如空格,回車和換行)就會提示出錯,提示信息如下:“Header had all ready send by”!。
在PHP4.0里面加入了緩沖區控制的幾個函數.
函數名稱ob_start
函數格式void ob_start(void)
功能:打開輸出緩沖區。
說明:當緩沖區激活時,所有來自PHP程序的非文件頭信息均不會發送,而是保存在內部緩沖區。為了輸出緩沖區的內容,可以使用ob_end_flush()或者使用ob_end_clean()來輸出緩沖區的內容。
實例分析:
1、用緩沖區控制的函數防止文件頭發送信息出錯。
<? //PHP提示符
ob_start(); //打開緩沖區
echo "Welcome /n"; //輸出
header("location:next.php"); //把瀏覽器重定向到next.php
?>
如果去掉ob_start,PHP就會提示在文件的第4行出錯,出錯信息為“Header had all ready send by”,但是加上ob_start,就不會提示出錯,原因是當打開了緩沖區,echo后面的字符不會輸出到瀏覽器,而是保留在服務器的緩沖區中,直到你使用flush或者ob_end_flush才會輸出,所以并不會出現文件頭已輸出的錯誤!
PHP4.0.4有一個新的輸出緩存句柄ob_gzhandler,它與前面的類相似,但用法不同。使用ob_gzhandler時要在php.ini中加入的內容如下:
output_handler = ob_gzhandler ;
這行代碼使得PHP激活輸出緩存,并壓縮它發送出去的所有內容。如果由于某種原因你不想在php.ini中加上這行代碼,你還可以通過PHP源文件所在目錄的.htaccess文件改變默認的服務器行為(不壓縮),語法如下:
php_value output_handler ob_gzhandler
或者是從PHP代碼調用,如下所示:
ob_start("ob_gzhandler");
采用輸出緩存句柄的方法確實非常有效,而且不會給服務器帶來什么特殊的負荷。但必須注意的是,Netscape Communicator對壓縮圖形的支持不佳,因此除非你能夠保證所有用戶都使用IE瀏覽器,否則你應該禁止壓縮JPEG和GIF圖形。一般地,對于所有其他文件,這種壓縮都有效,但建議你針對各種瀏覽器都分別進行測試,特別是當你使用了特殊的插件或者數據查看器時這一點尤其重要。
使用前面介紹的各種技術,你能夠顯著地改善網站的性能表現,但應該注意的是: PHP可能是、也可能不是性能瓶頸所在。務必仔細地觀察每一個和應用性能有關的因素,比如數據庫等。
單純使用本文技術只能在一定限度之內提高Web服務器的性能。因此在歸咎于PHP以及它的緩存之前,不妨看看是否應該升級服務器以及是否可以引入負載平衡技術(后者需要較大的投資)。
不要低估內容壓縮的作用。雖然你在100 MB/s的LAN連接下看到Web應用響應非常迅速,但使用Modem連接的用戶不會,他們只會抱怨你那100 Kb的HTML頁面實在過于龐大。
相關函數簡介:
1、Flush:刷新緩沖區的內容,輸出。
函數格式:flush()
說明:這個函數經常使用,效率很高。
2、ob_start :打開輸出緩沖區
函數格式:void ob_start(void)
說明:當緩沖區激活時,所有來自PHP程序的非文件頭信息均不會發送,而是保存在內部緩沖區。
為了輸出緩沖區的內容,可以使用ob_end_flush()或flush()輸出緩沖區的內容。
3 、ob_get_contents :返回內部緩沖區的內容。
使用方法:string ob_get_contents(void)
說明:這個函數會返回當前緩沖區中的內容,如果輸出緩沖區沒有激活,則返回 FALSE 。
4、ob_get_length:返回內部緩沖區的長度。
使用方法:int ob_get_length(void)
說明:這個函數會返回當前緩沖區中的長度;和ob_get_contents一樣,如果輸出緩沖區沒有激
活。則返回 FALSE。
5、ob_end_flush :發送內部緩沖區的內容到瀏覽器,并且關閉輸出緩沖區。
使用方法:void ob_end_flush(void)
說明:這個函數發送輸出緩沖區的內容(如果有的話)。
6、ob_end_clean:刪除內部緩沖區的內容,并且關閉內部緩沖區
使用方法:void ob_end_clean(void)
說明:這個函數不會輸出內部緩沖區的內容而是把它刪除!
7、ob_implicit_flush:打開或關閉絕對刷新
使用方法:void ob_implicit_flush ([int flag])
說明:使用過Perl的人都知道$│=x的意義,這個字符串可以打開/關閉緩沖區,而ob_implicit_flush函數也和那個一樣,默認為關閉緩沖區,打開絕對輸出后,每個腳本輸出都直接發送到瀏覽器,不再需要調用 flush()
二、深入了解:
1. 關于Flush函數:
這個函數在PHP3中就出現了,是一個效率很高的函數,他有一個非常有用的功能就是刷新browser的cache.我們舉一個運行效果非常明顯的例子來說明flush.