控制PHP的输出:缓存并压缩动态页面


Posted in PHP onJune 11, 2013

mod_gzip是一个Apache模块,其功能是使用Gzip压缩静态的html页面,遵循IETF标准的浏览器可以接受gzip编码(IE, Netscape等)。mod_gzip可以将页面的下载时间提高4-5倍。我强烈建议你在你的web服务器上使用mod_gzip。然而,我们还必须用PHP建立我们自己的压缩引擎。在这篇文章里,我将要介绍如何使用PHP的输出控制函数来大幅加速页面载入速度。

介绍PHP的输出控制函数

PHP4中最令人满意的事是——你可以让PHP缓存所有由脚本生成的输出,在你决定把它们送出之前,浏览器方是不会收到任何内容的。在脚本程序中你能用这些函数来设置header、cookies,然而这只是强大的输出函数的一小部分功能。

<?php 
void ob_start(void); 
?>

告诉PHP处理器把所有的输出重定向到一个内部的缓存(buffer)中。在调用ob_start之前,没有输出会被送到浏览器去。

<?php 
string ob_get_contents(void); 
?>

该函数将“输出缓存”(output buffer)以字符串的形式返回。你可以调用该函数把积累下来的输出送到浏览器中。(仅在把buffering功能关闭之后!!)

<?php 
int ob_get_length(void); 
?>

返回缓存中的字符串的长度。

<?php 
void ob_end_clean(void); 
?>

清空输出缓存,并将输出缓存关闭。在缓存中的内容输出到浏览器之前,必须使用这个函数。
void 501([int flag])
用来打开/关闭隐含的flush动作开关(默认是关)。如果flush是开的,每次调用print/echo或是其它输出命令的时候,输出的内容会被立即送到浏览器端。

使用输出控制来压缩PHP输出
你必须使用PHP4里编译的Zlib扩展包来压缩输出。如果需要的话,可以查看PHP文档中有关Zlib包的安装指导。
首先,初始化输出缓存:

<?php 
ob_start(); 
ob_implicit_flush(0); 
?>

之后,用print, echo, 或其他你喜欢的方法生成所有输出内容,例如:

<?php
print("Hey this is a compressed output!"); 
?>

页面生成后,我们取回输出内容:

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

之后,必须检测浏览器是否支持压缩数据。如果支持,浏览器会发给服务器端一个ACCEPT-ENCODEING HTTP头。我们只需检查$HTTP_ACCEPT_ENCODING变量中是否有“gzip,deflate”字串。

<?php 
if(ereg('gzip, deflate',$HTTP_ACCEPT_ENCODING)) { 
// 在这里生成 Gzip 压缩的内容 
} else { 
echo $contents; 
} 
?>

这种方法使用起来既简单又结构清晰。下面让我们看看如何生成压缩的输出:

<?php 
//告诉浏览器将要收到的是gzip数据 
//当然在此之前,你已经检查了它们是否支持gzip,x-gzip数据格式 
//如果支持的是x-gzip,那么下面的头就要用z-gzip来代替 
header("Content-Encoding: gzip"); 
//显示gzip文件的头 
//只需显示一次 
echo "x1fx8bx08x00x00x00x00x00"; 
//计算出文件的大小和CRC码 
$Size = strlen($contents); 
$Crc = crc32($contents); 
//压缩数据 
$contents = gzcompress($contents, 9); 
//我们不能就这样输出,因为CRC码是混乱的。 
//如果我在这里使用“echo $contents”,压缩的数据会被送出, 
//但是却不完整。文件最后的四个字节是CRC校验码,可是只发出去了三个字节。 
//最后一个字节被丢掉了。我不知道这个bug在4.0.2版中解决了没有, 
//不过最好避免错误的方法是把正确的CRC校验码加到压缩的数据的末尾。 
// 
//把旧的CRC校验码剥离 
$contents = substr($contents, 0, strlen($contents) - 4); 
//仅显示压缩的数据 
echo $contents; 
//输出CRC,和原来数据的大小(字节) 
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 
ob_start(); 
ob_implicit_flush(0); 
print("I'm compressed!n"); 
$contents = ob_get_contents(); 
ob_end_clean(); 
header("Content-Encoding: gzip"); 
echo "x1fx8bx08x00x00x00x00x00"; 
$Size = strlen($contents); 
$Crc = crc32($contents); 
$contents = gzcompress($contents, 9); 
$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输出

当PHP4还没问世,我不得不使用PHP3的时候,我对开发一些缓存机制来减少数据库的载入、对文件系统的存取十分感兴趣。在PHP3中没有什么特别好的方法,但是有了输出缓存之后,在PHP4中一切变得容易多了。
这有一个简单的例子:

<?php 
//为请求的URI构造一个文件名 
$cached_file=md5($REQUEST_URI); 
if((!file_exists("/cache/$cached_file"))||(!is_valid("/cache/$cached_file"))) { 
//is_valid函数验证缓存,你可以用这个函数检查Cache是否过期或其他特定的条件。 
//如果文件不在Cache中或者不可用则生成输出 
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输出控制函数对把脚本生成的输出重定向到缓存中十分有用。为支持gzip的浏览器输出压过的缓存数据可以减少载入时间。也可作为缓存机制来减少对数据源的存取(数据库或文件),这对使用XML意义重大。
如果我们用PHP建立一个引擎,缓存从数据源得到的数据(xml文档和数据库),并且动态的生成XML格式的内容(没有外观-presentation)我们可以得到这些XML的输出,并使用XSLT转换成任意一种我们想要的外观格式(html, wap, palm, pdf等)。使用PHP4的输出缓存和Sablotron XSLT扩展可以很好地完成这个任务。

PHP 相关文章推荐
php反射应用示例
Feb 25 PHP
php实现获取及设置用户访问页面语言类
Sep 24 PHP
AJAX PHP无刷新form表单提交的简单实现(推荐)
Sep 09 PHP
thinkPHP模板中函数的使用方法示例
Nov 30 PHP
PHP实现对xml进行简单的增删改查(CRUD)操作示例
May 19 PHP
Windows平台实现PHP连接SQL Server2008的方法
Jul 26 PHP
php中pcntl_fork创建子进程的方法实例
Mar 14 PHP
Laravel 默认邮箱登录改成用户名登录的实现方法
Aug 12 PHP
php校验公钥是否可用的实例方法
Sep 17 PHP
关于Laravel参数验证的一些疑与惑
Nov 19 PHP
php 使用ActiveMQ发送消息,与处理消息操作示例
Feb 23 PHP
PHP如何开启Opcache功能提升程序处理效率
Apr 27 PHP
基于PHP导出Excel的小经验 完美解决乱码问题
Jun 10 #PHP
win7+apache+php+mysql环境配置操作详解
Jun 10 #PHP
浅谈php中mysql与mysqli的区别分析
Jun 10 #PHP
探讨php中防止SQL注入最好的方法是什么
Jun 10 #PHP
php防注入,表单提交值转义的实现详解
Jun 10 #PHP
PHP获取当前页面完整URL的实现代码
Jun 10 #PHP
如何判断php数组的维度
Jun 10 #PHP
You might like
php读取30天之内的根据算法排序的代码
2008/04/06 PHP
PHP下用rmdir实现删除目录的三种方法小结
2008/04/20 PHP
PHP 中关于ord($str)&amp;gt;0x80的详细说明
2012/09/23 PHP
php常用字符串长度函数strlen()与mb_strlen()用法实例分析
2019/06/25 PHP
JavaScript学习笔记(一) js基本语法
2011/10/25 Javascript
推荐40个非常优秀的jQuery插件和教程【系列三】
2011/11/09 Javascript
jQuery结合PHP+MySQL实现二级联动下拉列表[实例]
2011/11/15 Javascript
jquery获取html元素的绝对位置和相对位置的方法
2014/06/20 Javascript
JavaScript通过Date-Mask将日期转换成字符串的方法
2015/06/04 Javascript
jQuery根据用户电脑是mac还是pc加载对应样式的方法
2015/06/26 Javascript
Jquery插件之Fancybox丰富的弹出层效果附源码下载
2015/12/02 Javascript
jQuery遮罩层实现方法实例详解(附遮罩层插件)
2015/12/08 Javascript
jquery实现的判断倒计时是否结束代码
2016/02/05 Javascript
判断js的Array和Object的实现方法
2016/08/29 Javascript
setTimeout函数的神奇使用
2017/02/26 Javascript
ES6使用let命令更简单的实现块级作用域实例分析
2017/03/31 Javascript
Angular实现的自定义模糊查询、排序及三角箭头标注功能示例
2017/12/28 Javascript
玩转Koa之koa-router原理解析
2018/12/29 Javascript
nuxt配置通过指定IP和端口访问的实现
2020/01/08 Javascript
通过实例解析javascript Date对象属性及方法
2020/11/04 Javascript
[59:59]EG vs IG 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
[06:59]DOTA2-DPC中国联赛3月7日Recap集锦
2021/03/11 DOTA
python 运算符 供重载参考
2009/06/11 Python
Python对象体系深入分析
2014/10/28 Python
python 中random模块的常用方法总结
2017/07/08 Python
Python2中文处理纪要的实现方法
2018/03/10 Python
如何在vscode中安装python库的方法步骤
2021/01/06 Python
高清屏下canvas重置尺寸引发的问题的解决
2019/10/14 HTML / CSS
Kneipp克奈圃美国官网:德国百年精油配方的传承
2018/02/07 全球购物
英国领先的维生素和补充剂品牌:Higher Nature
2019/08/26 全球购物
初中三年学生的学习自我评价
2013/11/13 职场文书
幼儿园庆六一活动方案
2014/03/06 职场文书
2015年感恩父亲节演讲稿
2015/03/19 职场文书
公司仓管员岗位职责
2015/04/01 职场文书
实习证明格式范文
2015/06/16 职场文书
个人工作总结(管理人员)范文
2019/08/13 职场文书