Kenny 2007-8-1 22:25
关于输出缓冲的讨论(Output buffering)
<p>#########################################<br>##标题 关于输出缓冲的讨论<br>##整理 Diego Lynn @ 林全国<br>##原作 By Zeev Suraski <br>#########################################<br><br>关于输出缓冲的讨论(Output buffering)<br><br>目录<br><br>HTTP Header<br>为什么要使用Output Buffering技术<br>Output Buffering的<a href="http://www.phpchina.com/javascript:;" onClick="javascript:tagshow(event, '%B9%A4%D7%F7');" target="_self"><u><strong>工作</strong></u></a>原理<br>基本用法<br>高级用法<br>使事情更为简单<br>哈哈,我成功了<br><br>我个人认为,Output buffering是比较纯粹的<a href="http://www.phpchina.com/javascript:;" onClick="javascript:tagshow(event, 'PHP');" target="_self"><u><strong>PHP</strong></u></a> 4.0特征。尽管从概念上看来相当简单,但是output buffering功能非常强大,能使开发者更容易地开发高级而有效的程序。<br><br>本文将介绍HTTP header,以及output buffering如何帮助您处理HTTP header,并介绍了output buffering的一些高级用法。<br><br>HTTP Header[HTTP 标题]<br>对于使用HTTP 协议建立的每个请求,<a class="channel_keylink" href="http://edu.itbulo.com/server/Web/" target="_blank">Web服务器</a>产生的响应通常包括两个部分 – 标题和主体。例如,如果在<a class="channel_keylink" href="http://edu.itbulo.com/server/Web/" target="_blank">Web服务器</a>的文档根目录下有一个小文本文件,叫做example.txt,文件中包含文本Hello, world!,那么对此文件的HTTP 请求响应如下所示:<br><br>HTTP/1.1 200 OK <br>Date: Sat, 02 Sep 2000 21:40:08 GMT <br>Server: Apache/1.3.11 (Unix) mod_macro/1.1.1 PHP/4.0.2-dev <br>Last-Modified: Sat, 02 Sep 2000 21:39:49 GMT <br>ETag: "12600b-e-39b173a5" <br>Accept-Ranges: bytes <br>Content-Length: 14 <br>Connection: close <br>Content-Type: text/plain <br><br>Hello, world!<br><br>这个请求中的第一部分(就是较多的那部分)就是HTTP header。虽然用户在浏览器中看不到HTTP header,但它包含了用于浏览器的信息,例如文档内容类型,使用的协议版本,文档的最后更改日期等等。HTTP header并没有太多的规则,通常情况下,它的格式如下:<br><br>Field: Value[字段:值]<br>必须用空行将它们和文档主体分开。<br><br>可以从PHP脚本添加或更改此HTTP header的信息。例如,可以使用 header() 函数:<br><br>header("Location: http://www.php.net/"); // 重定向到 http://www.php.net/<br><br>也可以使用 SetCookie() 函数:<br><br>SetCookie("foo", "bar"); <br><br>你可能会知道HTTP cookies是使用 HTTP headers 来实现的。例如,以下PHP文件的 HTTP 请求响应<br><br><?php <br><br>SetCookie("foo", "bar"); <br><br>print "Set cookie."; <br><br>?> <br><br>将会是这样的:<br><br>HTTP/1.1 200 OK <br>Date: Sat, 02 Sep 2000 21:43:02 GMT <br>Server: Apache/1.3.11 (Unix) mod_macro/1.1.1 PHP/4.0.2-dev <br>X-Powered-By: PHP/4.0.2-dev <br>Set-Cookie: foo=bar <br>Connection: close <br>Content-Type: text/html <br>Set cookie.<br><br><br><br>浏览器读取从服务器返回的 HTTP header,知道送来了一个叫做 foo 的 cookie (在这里是一个 session cookie),它的值是 bar。<br><br>为什么要使用Output Buffering技术<br><br>早在PHP/FI 2.0时就很明显需要output buffering技术了。如果你使用过这种版本的PHP,那么可能还记得经常会碰到 Oops, SetCookie called after header has been sent 这个错误消息,并使你捎头抓耳,也弄不清是什么原因。<br><br>如果你已使用过PHP的最新版本 -- PHP 3.0 甚至 PHP 4.0 -- 那么你会知道这个错误消息: Oops, php_set_cookie called after header has been sent。或者,你在试图调用 PHP 的 header() 函数时会遇到 Cannot add header information - headers already sent 消息。一般来说,output buffering技术用户避免这些烦人的错误消息,同时开发人员也可用于高级的用途。<br><br>这些错误是什么时候产生的呢?如果你在已经发送了HTTP header之后试图添加或修改标题信息,以及在文档主体和标题之间缺少空行时,就会产生这些错误消息。为了理解这是如何产生的,让我们来看看PHP是如何处理HTTP header输出和主体输出的。<br><br>脚本开始执行时,它可以同时发送header(标题)信息和主体信息。 Header信息(来自 header() 或 SetCookie() 函数)并不会立即发送,相反,它被保存到一个列表中。 这样就可以允许你修改标题信息,包括缺省的标题(例如 Content-Type 标题)。但是,一旦脚本发送了任何非标题的输出(例如,使用 <a href="http://www.phpchina.com/javascript:;" onClick="javascript:tagshow(event, 'HTML');" target="_self"><u><strong>HTML</strong></u></a> <a href="http://www.phpchina.com/javascript:;" onClick="javascript:tagshow(event, '%B4%FA%C2%EB');" target="_self"><u><strong>代码</strong></u></a>块或 print() 调用),那么PHP就必须先发送所有的标题,然后再送出空行,终止 HTTP header,而在此之后才会继续发送主体数据。从这时开始,任何添加或修改标题信息的试图都是不允许的,并会发送上述的错误消息之一。<br><br>虽然这并不会引起多大的问题,有时候只是在发出任何输入之前终止HTTP header,从而引起脚本逻辑的复杂化而已。Output buffering技术可以解决这些问题。<br><br>Output Buffering的工作原理<br>启用output buffering时,在脚本发送输出时,PHP并不发送HTTP header。相反,它将此输出通过管道(pipe)输入到动态增加的缓存中(只能在PHP 4.0中使用,它具有中央化的输出机制)。你仍然可以修改,添加标题行,或者设置cookie,因为标题实际上并没有发送。最简单的情况是,当脚本终止时,PHP将自动发送HTTP header到浏览器,然后再发送输出缓冲中的内容。这简单吧。<br><br>基本用法<br><br>可以使用下面的四个函数,它们可以帮助你控制output buffering:<br><br>ob_start()<br>启用output buffering机制。 Output buffering支持多层次 -- 例如,可以多次调用 ob_start() 函数。<br><br>ob_end_flush()<br>发送output buffer(输出缓冲)并禁用output buffering机制。<br><br>ob_end_clean()<br>清除output buffer但不发送,并禁用output buffering。<br><br>ob_get_contents()<br>将当前的output buffer返回成一个字符串。允许你处理脚本发出的任何输出。<br><br><br><br>此外,可以启用 php.ini 中的 output_buffering 指令。如果启用了此指令,那么每个PHP脚本都相当于一开始就调用了ob_start()函数。<br><br>Example 1<br><br><?php ob_start(); ?> <br><br><h1>Example 1</h1> <br><br><?php <br><br>print "Hello, $user\n"; <br><br>SetCookie("Wow", "This cookie has been set even though we've already emitted output!"); <br><br>?> <br>这里,尽管你已发送了输出(HTML 代码块中和 print 语句中),也可以使用 SetCookie() 调用,而不会出错,真的要感谢output buffering机制。请注意使用output buffering机制用于这种目的会引起一定程度上的性能损失,因此最好缺省情况下不要启用此机制。但是,对于复杂一些的脚本,output buffering可以简化逻辑性。<br><br>Example 2<br><br><?php <br>ob_start(); <br>print "Here's a pretty dumb way to calculate the length of a string."; <br>$length = strlen(ob_get_buffer()); <br>ob_end_clean(); <br>?> <br><br>这个例子显示了一个效率很低的确定字符串长度的<a href="http://www.phpchina.com/javascript:;" onClick="javascript:tagshow(event, '%B7%BD%B7%A8');" target="_self"><u><strong>方法</strong></u></a>。它不是简单的使用strlen()函数处理,而是先启用 output buffering 机制,将字符串打印出来,然后再确定output buffer的长度。最后清除output buffer(并没有发送),然后禁用output buffering机制。<div>