PHP 输出缓存详解
前端之家收集整理的这篇文章主要介绍了
PHP 输出缓存详解,
前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
输出控制函数不对使用 header() 或 setcookie(),发送的文件头信息产生影响,只对那些类似于 echo() 和 PHP 代码的数据块有作用。
我们先举一个简单的例子,让大家对Output Control有一个大致的印象:
Example 1.
<div class="codetitle"><a style="CURSOR: pointer" data="56713" class="copybut" id="copybut56713" onclick="doCopy('code56713')"> 代码如下:
<div class="codebody" id="code56713">
<?
PHP ob_start(); //打开缓冲区
echo \"Hellon\"; //
输出 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才会
输出,所以并不会有任何
文件头
输出的
错误!
一、 相关函数简介:
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.
Example 2.
<div class="codetitle">
<a style="CURSOR: pointer" data="48427" class="copybut" id="copybut48427" onclick="doCopy('code48427')"> 代码如下: <div class="codebody" id="code48427">
<?
PHP for($i = 1; $i <= 300; $i++ ) print(“ “);
// 这一句话非常关键,cache的结构使得它的
内容只有达到一定的大小才能从浏览器里
输出 // 换言之,如果cache的
内容不达到一定的大小,它是不会在程序执行完毕前
输出的。经
// 过测试,我发现这个大小的底限是256个字符长。这意味着cache以后接收的
内容都会
// 源源不断的被发送出去。
For($j = 1; $j <= 20; $j++) {
echo $j.”
”;
flush(); //这一部会使cache新增的
内容被挤出去,
显示到浏览器上
sleep(1); //让程序“睡”一秒钟,会让你把
效果看得更清楚
}
?>
具体
效果你可以到这里看看http://www.
PHP2000.com/~uchinaboy/out.
PHP PHP2000的最新的
PHP聊天室就是用的这个技术,可惜的是源
代码未公开 L
注:如果在程序的首部加入ob_implicit_flush()打开绝对刷新,就可以在程序中不再使用flush(),这样做的好处是:提高效率!
2. 关于ob系列函数:
我想先引用我的好朋友y10k的一个例子:
Example 3.
比如你用得到服务器和客户端的设置信息,但是这个信息会因为客户端的不同而不同,如果想要保存
PHPinfo()
函数的
输出怎么办呢?在没有缓冲区控制之前,可以说一点办法也没有,但是有了缓冲区的控制,我们可以轻松的
解决:
<div class="codetitle">
<a style="CURSOR: pointer" data="26401" class="copybut" id="copybut26401" onclick="doCopy('code26401')"> 代码如下: <div class="codebody" id="code26401">
<?
PHP 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信息保存下来,这在以前恐怕没有办法办到!其实上面就是将一些“过程”转化为“
函数”的
方法!
或许有人会问:“难道就这个样子吗?还有没有其他用途?”当然有了,比如笔者论坛的
PHP 语法加亮
显示就和这个有关(
PHP默认的语法加亮
显示函数会直接
输出,不能保存结果,如果在每次
调用都
显示恐怕会很浪费
cpu,笔者的论坛就把语法加亮
函数显示的结果用控制缓冲区的
方法保留了),大家如果感兴趣的话可以来看看http://www.z
PHP.com/bbs/!
可能现在大家对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系列
函数实现。
对于第一种
方法,因为不是这篇
文章所要研究的问题,所以不再赘述。
我们现在来看一看第二种
方法的具体实现:
Example 4.
<?
PHP ob_start();//打开缓冲区
?>
PHP页面的全部
输出 <?
$content = ob_get_contents();//取得
PHP页面输出的全部
内容 $fp = fopen(“output00001.html”,“w”); //创建一个
文件,并打开,准备写入
fwrite($fp,$content); //把
PHP页面的
内容全部写入output00001.html,然后……
fclose($fp);
?>
这样,所谓的静态模版就很容易的被实现了……
二、 捕捉
输出 以上的Example 4.是一种最简单的情况,你还可以在写入前对$content进行操作……
你可以设法捕捉一些关键字,然后去对它进行再处理,比如Example 3.所述的
PHP语法高亮
显示。个人认为,这个
功能是此
函数最大的精华所在,它可以
解决各种各样的问题,但需要你有足够的想象力……
Example 5.
<?
Function run_code($code) {
If($code) {
ob_start();
eval($code);
$contents = ob_get_contents();
ob_end_clean();
}else {
echo “
错误!没有
输出”;
exit();
}
return $contents;
}
以上这个例子的用途不是很大,不过很典型$code的本身就是一个含有变量的
输出页面,而这个例子用eval把$code中的变量替换,然后对
输出结果再进行
输出捕捉,再一次的进行处理……
Example 6. 加快传输
<?
/*
Title.........: PHP4 HTTP Compression Speeds up the Web
Version.......: 1.20
Author........: catoc catoc@163.net
Filename......: gzdoc.
PHP Last changed..: 18/10/2000
Requirments...:
PHP4 >= 4.0.1
PHP was configured with --with-zlib[=DIR]
Notes.........: Dynamic Content Acceleration compresses
the data transmission data on the fly
code by sun jin hu (catoc)
catoc@163.net Most newer browsers since 1998/1999 have
been equipped to support the HTTP 1.1
standard known as \"content-encoding.\"
Essentially the browser indicates to the
server that it can accept \"content encoding\"
and if the server is capable it will then
compress the data and transmit it. The
browser decompresses it and then renders
the page.
Modified by John Lim (jlim@natsoft.com.my)
based on ideas by Sandy McArthur,Jr
Usage........:
No space before the beginning of the first \'<?\' tag.
------------Start of file----------
|<?
| include(\'gzdoc.PHP\');
|? >
|
|... the page ...
|
|<?
| gzdocout();
|? >
*
-------------End of file-----------
/
ob_start();
ob_implicit_flush(0);
function CheckCanGzip(){
global $HTTP_ACCEPT_ENCODING;
if (headers_sent() || connection_timeout() || connection_aborted()){
return 0;
}
if (strpos($HTTP_ACCEPT_ENCODING,\'x-gzip\') !== false) return \"x-gzip\";
if (strpos($HTTP_ACCEPT_ENCODING,\'gzip\') !== false) return \"gzip\";
return 0;
}
/
$level = compression level 0-9,0=none,9=max /
function GzDocOut($level=1,$debug=0){
$ENCODING = CheckCanGzip();
if ($ENCODING){
print \"nn\";
$Contents = ob_get_contents();
ob_end_clean();
if ($debug){
$s = \"
Not compress length: \".strlen($Contents);
$s .= \"
Compressed length: \".strlen(gzcompress($Contents,$level));
$Contents .= $s;
}
header(\"Content-Encoding: $ENCODING\");
print \"x1fx8bx08x00x00x00x00x00\";
$Size = strlen($Contents);
$Crc = crc32($Contents);
$Contents = gzcompress($Contents,$level);
$Contents = substr($Contents,strlen($Contents) - 4);
print $Contents;
print pack(\'V\',$Crc);
print pack(\'V\',$Size);
exit;
}else{
ob_end_flush();
exit;
}
}
?>
这是catoc的一段很早以前的代码,是在weblogs.com看到的,他利用了zlib的函数,对传输的内容进行了压缩,测试表明,对于10k以上的页面,会产生效果,而且页面越大,效果越明显……
原文链接:https://www.f2er.com/php/29166.html