PHP缓冲机制:缓冲控制,压缩http响应数据,缓冲 PHP输出

林涛 2383℃ 0评论

动态网站的内容加速显示十分重要,本文通过对 PHP 几个函数的深入讨论,提出了 PHP 网页压缩和缓冲的解决方案

一、介绍几个控制 PHP 输出的函数

PHP采用了缓冲机制,在你决定发送以前,所有内容只是存在于缓冲中,而不是直接发送给浏览器,虽然你可以用 header 和 setcookie 函数来实现,但是这两个函数相比于功能强大的输出函数来说只是一点“雕虫小技”。让我们来看看这些函数的真本事:

void ob_start(void);

本函数告诉 PHP 处理器把所有输出重定向到内部缓冲,调用这个函数后,就不会有输出到浏览器。

string ob_get_contents(void);

本函数把输出缓冲返回到一个字符串,你可以用来把堆积起来的输出一起发送到浏览器。当然要先关掉缓冲。

int ob_get_length(void);

本函数返回输出缓冲的长度。

void ob_end_clean(void);

本函数清除并关闭缓冲。在输出到浏览器之前你需要使用这个函数。

void ob_implicit_flush ([int flag])

本函数用来控制隐式缓冲泻出,缺省为 off,如果打开时,对每个 print/echo 或者输出命令的结果都发送到浏览器。

二、采用输出控制来压缩 PHP 的输出

在开始之前,要保证你的 PHP4 编译时支持 Zlib。
首先,初始化输出缓冲:

ob_start();
ob_implicit_flush(0);
?>

然后产生所有的输出内容。

print(”本例为压缩输出!”);
?>
页面生成以后,采用:

$contents = ob_get_contents();
ob_end_clean();
?>

还要检查浏览器是否支持压缩数据,我们采用在变量 $HTTP_ACCEPT_ENCODING 中检查 “gzip, deflate”的办法:

if(ereg(’gzip, deflate’,$HTTP_ACCEPT_ENCODING)) {
// 产生 gzip 后的内容
} else {
echo $contents;
}
?>

下面我们分析怎样产生 gzip 输出:

// 告诉浏览器下面接收的是 gzip 数据。
header(”Content-Encoding: gzip”);
// 显示 gzip 文件的文件头
// 只要一次就够了
echo “x1fx8bx08×00×00×00×00×00″;
// 计算长度和 CRC 校验码
$Size = strlen($contents);
$Crc = crc32($contents);
// 压缩数据
$contents = gzcompress($contents, 9);
// 不能直接在这里输出内容,因为还没有写入 CRC 呢!
$contents = substr($contents, 0, strlen($contents) – 4);
echo $contents;
gzip_PrintFourChars($Crc);
gzip_PrintFourChars($Size);
function gzip_PrintFourChars($Val) {
for ($i = 0; $i < 4; $i ++) { echo chr($Val % 256); $Val = floor($Val / 256); } } ?>

三、缓冲 PHP 的输出

在 PHP里能很容易的实现缓冲,我们来看例子:

// 对请求的 URI 产生唯一的文件名。
$cached_file=md5($REQUEST_URI);
if((!file_exists(”/cache/$cached_file”))||(!is_valid(”/cache/$cached_file”))) {
ob_start();
ob_implicit_flush(0);
// 在这里输出缓冲
$contents = ob_get_contents();
ob_end_clean();
$fil=fopen($cached_file,”w+”);
fwrite($fil,$contents,$strlen($contents));
fclose($fil);
}

readfile($cached_file);
?>

四、结论

PHP 输出缓冲函数在操作脚本输出方面十分有用,把缓冲压缩后输出能减少 80% 的输出时间,这对于存取其他数据资源(例如数据库或者 XML)来说,也是一个很好的缓冲机制。

PHP设置HTTP响应CODE

header()函数的作用是:发送一个原始 HTTP 标头[Http Header]到客户端。
标头 (header) 是服务器以 HTTP 协义传 HTML 资料到浏览器前所送出的字串,在标头
与 HTML 文件之间尚需空一行分隔。有关 HTTP 的详细说明,可以参 RFC 2068 官方文件
(http://www.w3.org/Protocols/rfc2068/rfc2068)。
在 PHP 中送回 HTML 资料前,需先 传完所有的标头。

使用范例

范例一: 本例使浏览器重定向到 PHP 的官方网站。

<?PHP
Header(“Location: http://www.php.net”;);
exit;   //在每个重定向之后都必须加上“exit”,避免发生错误后,继续执行。
?>
<?php
/**
@title:PHP定时跳转
@功能:等待指定的时间,然后再跳转到指定页面(代替html meta方式)
*/
header(“refresh:3;url=http://axgle.za.net”);
print(‘正在加载,请稍等…<br>三秒后自动跳转~~~’);
/*
补充说明:
若等待时间为0,则与header(“location:”)等效。
*/
header重定向 就等价于替用户在地址栏输入url
?>

范例二:禁止页面在IE中缓存

要使用者每次都能得到最新的资料,而不是 Proxy 或 cache 中的资料,可以使用下列的标头

<?PHP
header
( 'Expires: Mon, 26 Jul 1997 05:00:00 GMT'
);
header( 'Last-Modified: ' . gmdate( 'D, d M Y H:i:s' ) . ' GMT'
);
header( 'Cache-Control: no-store, no-cache, must-revalidate'
);
header( 'Cache-Control: post-check=0, pre-check=0', false
);
header( 'Pragma: no-cache' );
//兼容http1.0和https
?>

CacheControl = no-cache
Pragma=no-cache
Expires = -1

Expires是个好东东,如果服务器上的网页经常变化,就把它设置为-1,表示立即过期。如果一个网页每天凌晨1点更新,可以把Expires设置为第二天的凌晨1点。
当HTTP1.1服务器指定CacheControl = no-cache时,浏览器就不会缓存该网页。

旧式 HTTP 1.0 服务器不能使用 Cache-Control 标题。所以为了向后兼容 HTTP 1.0 服务器,IE使用Pragma:no-cache 标题对 HTTP 提供特殊支持。
如果客户端通过安全连接 (https://) 与服务器通讯,且服务器在响应中返回 Pragma:no-cache 标题,则 Internet Explorer 不会缓存此响应。
注意:Pragma:no-cache 仅当在安全连接中使用时才防止缓存,如果在非安全页中使用,处理方式与 Expires:-1 相同,该页将被缓存,但被标记为立即过期。

http-equiv meta标记:

在html页面中可以用http-equiv meta来标记指定的http消息头部。老版本的IE可能不支持html meta标记,所以最好使用http消息头部来禁用缓存。

范例三: 让使用者的浏览器出现找不到档案的信息。

网上很多资料这样写:php的函数header()可以向浏览器发送Status标头,
如 header(”Status: 404 Not Found”)。

但是我发现实际上浏览器返回的响应却是:

HTTP/1.x 200 OK
Date: Thu, 03 Aug 2006 07:49:11 GMT
Server: Apache/2.0.55 (Win32) PHP/5.0.5
X-Powered-By: PHP/5.0.5
Status: 404 Not Found
Content-Length: 0
Keep-Alive: timeout=15, max=98
Connection: Keep-Alive
Content-Type: text/html

查了一些资料,正确的写法是:
header(”http/1.1 404 Not Found”);

第一部分为HTTP协议的版本(HTTP-Version);第二部分为状态代码(Status);第三部分为原因短语(Reason-Phrase)。

范例四:让使用者下载档案( 隐藏文件的位置 )

html标签 就可以实现普通文件下载。如果为了保密文件,就不能把文件链接告诉别人,可以用header函数实现文件下载。
<?php
header(“Content-type: application/x-gzip”);
header(“Content-Disposition: attachment; filename=文件名\”);
header(“Content-Description: PHP3 Generated Data”);
?>

范例四:header函数前输入内容

一般来说在header函数前不能输出html内容,类似的还有setcookie() 和 session 函数,这些函数需要在输出流中增加消息头部信息。如果在header()执行之前有echo等语句,当后面遇到header()时,就会报出 “Warning: Cannot modify header information – headers already sent by ….”错误。就是说在这些函数的前面不能有任何文字、空行、回车等,而且最好在header()函数后加上exit()函数。例如下面的错误写法,在两个 php代码段之间有一个空行:

//some code here
?>
//这里应该是一个空行
header(”http/1.1 403 Forbidden”);
exit();
?>

原因是:PHP脚本开始执行 时,它可以同时发送http消息头部(标题)信息和主体信息. http消息头部(来自 header() 或 SetCookie() 函数)并不会立即发送,相反,它被保存到一个列表中. 这样就可以允许你修改标题信息,包括缺省的标题(例如 Content-Type 标题).但是,一旦脚本发送了任何非标题的输出(例如,使用 HTML 或 print() 调用),那么PHP就必须先发送完所有的Header,然后终止 HTTP header.而后继续发送主体数据.从这时开始,任何添加或修改Header信息的试图都是不允许的,并会发送上述的错误消息之一。

解决办法:

修改php.ini打开缓存(output_buffering),或者在程序中使用缓存函数ob_start(),ob_end_flush()等。原 理是:output_buffering被启用时,在脚本发送输出时,PHP并不发送HTTP header。相反,它将此输出通过管道(pipe)输入到动态增加的缓存中(只能在PHP 4.0中使用,它具有中央化的输出机制)。你仍然可以修改/添加header,或者设置cookie,因为header实际上并没有发送。当全部脚本终止 时,PHP将自动发送HTTP header到浏览器,然后再发送输出缓冲中的内容。

=================================================================
PHP 手册实例应用

1:您可以使用heder命令,强制使浏览器使用新鲜的内容(无缓存) 。
也可以给网址增加了一个唯一的编号,使其每次都读取新的内容,避免缓存。
example:
<?
print “<img src=’yourfile.jpg’>”;   //通常读取的是缓存文件
?>

<?
print “<img src=’yourfile.jpg?”.time().“‘>”;   //增加了唯一的编号,使浏览器重新请求

w//print
“<img src=’yourfile.jpg?”.rand(100,999).“‘>”;
?>

2: 下面是个很好的函数,将图片传送给浏览器显示。
<?php
function PE_img_by_path($PE_imgpath = “”)
{
if (file_exists($PE_imgpath)) {
$PE_imgarray = pathinfo($PE_imgpath);

$iconcontent = file_get_contents($PE_imgpath);
header(“Content-type: image/” . $PE_imgarray[“extension”]);
header(‘Content-length: ‘ . strlen($iconcontent));
echo $iconcontent;
die(0);
}
return false;
}
?>
更多的实例

  1. // ok
  2. header(‘HTTP/1.1 200 OK’);
  3. //设置一个404头:
  4. header(‘HTTP/1.1 404 Not Found’);
  5. //设置地址被永久的重定向
  6. header(‘HTTP/1.1 301 Moved Permanently’);
  7. //转到一个新地址
  8. header(‘Location: http://www.example.org/’);
  9. //文件延迟转向:
  10. header(‘Refresh: 10; url=http://www.example.org/’);
  11. print ‘You will be redirected in 10 seconds’;
  12. //当然,也可以使用html语法实现
  13. // <meta http-equiv=”refresh” content=”10;http://www.example.org/ />
  14. // override X-Powered-By: PHP:
  15. header(‘X-Powered-By: PHP/4.4.0’);
  16. header(‘X-Powered-By: Brain/0.6b’);
  17. //文档语言
  18. header(‘Content-language: en’);
  19. //告诉浏览器最后一次修改时间
  20. $time = time() – 60; // or filemtime($fn), etc
  21. header(‘Last-Modified: ‘.gmdate(‘D, d M Y H:i:s’, $time).’ GMT’);
  22. //告诉浏览器文档内容没有发生改变
  23. header(‘HTTP/1.1 304 Not Modified’);
  24. //设置内容长度
  25. header(‘Content-Length: 1234’);
  26. //设置为一个下载类型
  27. header(‘Content-Type: application/octet-stream’);
  28. header(‘Content-Disposition: attachment; filename=”example.zip”‘);
  29. header(‘Content-Transfer-Encoding: binary’);
  30. // load the file to send:
  31. readfile(‘example.zip’);
  32. // 对当前文档禁用缓存
  33. header(‘Cache-Control: no-cache, no-store, max-age=0, must-revalidate’);
  34. header(‘Expires: Mon, 26 Jul 1997 05:00:00 GMT’); // Date in the past
  35. header(‘Pragma: no-cache’);
  36. //设置内容类型:
  37. header(‘Content-Type: text/html; charset=iso-8859-1’);
  38. header(‘Content-Type: text/html; charset=utf-8’);
  39. header(‘Content-Type: text/plain’); //纯文本格式
  40. header(‘Content-Type: image/jpeg’); //JPG***
  41. header(‘Content-Type: application/zip’); // ZIP文件
  42. header(‘Content-Type: application/pdf’); // PDF文件
  43. header(‘Content-Type: audio/mpeg’); // 音频文件
  44. header(‘Content-Type: application/x-shockw**e-flash’); //Flash动画
  45. //显示登陆对话框
  46. header(‘HTTP/1.1 401 Unauthorized’);
  47. header(‘WWW-Authenticate: Basic realm=”Top Secret”‘);
  48. print ‘Text that will be displayed if the user hits cancel or ‘;
  49. print ‘enters wrong login data’;

如需转载请注明: 转载自26点的博客

本文链接地址: PHP缓冲机制:缓冲控制,压缩http响应数据,缓冲 PHP输出

转载请注明:26点的博客 » PHP缓冲机制:缓冲控制,压缩http响应数据,缓冲 PHP输出

喜欢 (0)
发表我的评论
取消评论

表情