搜索
首页后端开发php教程PHP基础教程 PHP的页面缓冲处理机制

PHP有很多机制、函数,其实就是魔术师,重复发挥好,其实甚至是简单应用,就会出现神奇的效果。兄弟连PHP培训 

这里来讲一个ob_start()函数。 

  ob_start()函数用于打开缓冲区,比如header()函数之 前如果就有输出,包括回车\空格\换行\都会有”Header had all ready send by”的错误,这时可以先用ob_start()打开缓冲区PHP代码的数据块和echo()输出都会进入缓冲区而不会立刻输出.当然打开缓冲区的作用很 多,只要发挥你的想象.可以总结以下四点: 

  1.用于header()之前 

  ob_start(); //打开缓冲区 

  echo \”Hellon\”; //输出 

  header(”location:index.php”);//把浏览器重定向到index.php 

  ob_end_flush();//输出全部内容到浏览器 

  ?> 

  2.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 

  ?> 

  3.静态页面技术 

  ob_start();//打开缓冲区 

  ?> 

  php页面的全部输出 

  $content =ob_get_contents();//取得php页面输出的全部内容 

  $fp =fopen(”output00001.html”, “w”); //创建一个文件,并打开,准备写入 

  fwrite($fp, $content); //把php页面的内容全部写入output00001.html,然后…… 

  fclose($fp); 

  ?> 

  4.输出代码 

  Function run_code($code) { 

  If($code) { 

  ob_start(); 

  eval($code); 

  $contents =ob_get_contents(); 

  ob_end_clean(); 

  }else { 

  echo “错误!没有输出”; 

  exit(); 

  } 

  return $contents; 

  } 

  Output Control 函数可以让你自由控制脚本中数据的输出。它非常地有用,特别是对于:当你想在数据已经输出后,再输出文件头的情况。输出控制函数不对使用 header() 或 setcookie(), 发送的文件头信息产生影响,只对那些类似于 echo() 和PHP 代码的数据块有作用。 

  我们先举一个简单的例子,让大家对OutputControl有一个大致的印象: 

  Example 1. 

  CODE
  ob_start(); //打开缓冲区 

  echo \”Hellon\”; //输出 

  header(”location:index.php”);//把浏览器重定向到index.php 

  ob_end_flush();//输出全部内容到浏览器 

  ?> 

  所有对header()函数有了解的人都知道,这个函数会发送一段文件头给浏览器,但是如果在使用这个函数之前已经有了任何输出(包括空输出,比如 空格, 回车和换行)就会提示出错。如果我们去掉第一行的ob_start(),再执行此程序,我们会发现得到了一条错误提示:”Header hadall 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:返回内部缓冲区的内容。 

  使用方法:stringob_get_contents(void) 

  说明:这个函数会返回当前缓冲区中的内容,如果输出缓冲区没有激活,则返回 FALSE 。 

  4、ob_get_length:返回内部缓冲区的长度。 

  使用方法:intob_get_length(void) 

  说明:这个函数会返回当前缓冲区中的长度;和ob_get_contents一样,如果输出缓冲区没有激活。则返回 FALSE。 

  5、ob_end_flush:发送内部缓冲区的内容到浏览器,并且关闭输出缓冲区。 

  使用方法:voidob_end_flush(void) 

  说明:这个函数发送输出缓冲区的内容(如果有的话)。 

  6、ob_end_clean:删除内部缓冲区的内容,并且关闭内部缓冲区 

  使用方法:voidob_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. 

  CODE
  for($i = 1; $i
  // 这一句话非常关键,cache的结构使得它的内容只有达到一定的大小才能从浏览器里输出 

  // 换言之,如果cache的内容不达到一定的大小,它是不会在程序执行完毕前输出的。经 

  // 过测试,我发现这个大小的底限是256个字符长。这意味着cache以后接收的内容都会 

  // 源源不断的被发送出去。 

  For($j = 1; $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()函数的输出怎么办呢?在没有缓冲区控制之前,可以说一点办法也没有,但是有了缓冲区的控制,我们可以轻松的解决: 

  CODE
  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.zphp.com/bbs/! 

  可能现在大家对ob_start()的功能有了一定的了解,上面的一个例子看似简单,但实际上已经掌握了使用ob_start()的要点。 

  .使用ob_start打开browser的cache,这样可以保证cache的内容在你调用flush(),ob_end_flush()(或程序执行完毕)之前不会被输出。 

  .现在的你应该知道你所拥有的优势:可以在任何输出内容后面使用header,setcookie以及session,这是ob_start一 个很大的特点;也可以使用ob_start的参数,在cache被写入后,然后自动运行命令,比如ob_start(\”ob_gzhandler\ “);而我们最常用的做法是用ob_get_contents()得到cache中的内容,然后再进行处理…… 

  .当处理完毕后,我们可以使用各种方法输出,flush(),ob_end_flush(),以及等到程序执行完毕后的自动输出。当然,如果你用的是ob_get_contents(),那么就要你自己控制输出方式了。 

  来,让我们看看能用ob系列函数做些什么…… 

  一、 静态模版技术 

  简介:所谓静态模版技术就是通过某种方式,使得用户在client端得到的是由PHP产生的html页面。如果这个html页面不会再被更新,那么当另外 的用户再次浏览此页面时,程序将不会再调用PHP以及相关的数据库,对于某些信息量比较大的网站,例如sina,163,sohu。类似这种的技术带来的 好处是非常巨大的。 

  我所知道的实现静态输出的有两种办法: 

  .通过y10k修改的phplib的一个叫template.inc.php类实现。 

  .使用ob系列函数实现。 

  对于第一种方法,因为不是这篇文章所要研究的问题,所以不再赘述。 

  我们现在来看一看第二种方法的具体实现: 

  Example 4. 

  CODE
  ob_start();//打开缓冲区 

  ?> 

  php页面的全部输出 

   

  $content =ob_get_contents();//取得php页面输出的全部内容 

  $fp =fopen(”output00001.html”, “w”); //创建一个文件,并打开,准备写入 

  fwrite($fp, $content); //把php页面的内容全部写入output00001.html,然后…… 

  fclose($fp); 

  ?> 

  这样,所谓的静态模版就很容易的被实现了…… 

  二、 捕捉输出 

  以上的Example 4.是一种最简单的情况,你还可以在写入前对$content进行操作…… 

  你可以设法捕捉一些关键字,然后去对它进行再处理,比如Example3.所述的PHP语法高亮显示。个人认为,这个功能是此函数最大的精华所在,它可以解决各种各样的问题,但需要你有足够的想象力…… 

  Example 5. 

  CODE 

  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. 加快传输 

  CODE 

  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; 

  } 

  functionGzDocOut($level=1,$debug=0){ 

  $ENCODING = CheckCanGzip(); 

  if ($ENCODING){ 

  print \”nn\”; 

  $Contents =ob_get_contents(); 

  ob_end_clean(); 

  if ($debug){ 

  $s = \”

Notcompress length: \”.strlen($Contents); 

  $s .= \” 

  Compressed length:\”.strlen(gzcompress($Contents,$level)); 

  $Contents .= $s; 

  } 

  header(\”Content-Encoding:$ENCODING\”); 

  print\”x1fx8bx08×00x00×00x00×00\”; 

  $Size = strlen($Contents); 

  $Crc = crc32($Contents); 

  $Contents =gzcompress($Contents,$level); 

  $Contents = substr($Contents,0, strlen($Contents) – 4); 

  print $Contents; 

  print pack(\’V\’,$Crc); 

  print pack(\’V\’,$Size); 

  exit; 

  }else{ 

  ob_end_flush(); 

  exit; 

  } 

  } 

  ?> 

  这是catoc的一段很早以前的代码,是在weblogs.com看到的,他利用了zlib的函数,对传输的内容进行了压缩,测试表明,对于10k以上的页面,会产生效果,而且页面越大,效果越明显……

声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
哪些常见问题会导致PHP会话失败?哪些常见问题会导致PHP会话失败?Apr 25, 2025 am 12:16 AM

PHPSession失效的原因包括配置错误、Cookie问题和Session过期。1.配置错误:检查并设置正确的session.save_path。2.Cookie问题:确保Cookie设置正确。3.Session过期:调整session.gc_maxlifetime值以延长会话时间。

您如何在PHP中调试与会话相关的问题?您如何在PHP中调试与会话相关的问题?Apr 25, 2025 am 12:12 AM

在PHP中调试会话问题的方法包括:1.检查会话是否正确启动;2.验证会话ID的传递;3.检查会话数据的存储和读取;4.查看服务器配置。通过输出会话ID和数据、查看会话文件内容等方法,可以有效诊断和解决会话相关的问题。

如果session_start()被多次调用会发生什么?如果session_start()被多次调用会发生什么?Apr 25, 2025 am 12:06 AM

多次调用session_start()会导致警告信息和可能的数据覆盖。1)PHP会发出警告,提示session已启动。2)可能导致session数据意外覆盖。3)使用session_status()检查session状态,避免重复调用。

您如何在PHP中配置会话寿命?您如何在PHP中配置会话寿命?Apr 25, 2025 am 12:05 AM

在PHP中配置会话生命周期可以通过设置session.gc_maxlifetime和session.cookie_lifetime来实现。1)session.gc_maxlifetime控制服务器端会话数据的存活时间,2)session.cookie_lifetime控制客户端cookie的生命周期,设置为0时cookie在浏览器关闭时过期。

使用数据库存储会话的优点是什么?使用数据库存储会话的优点是什么?Apr 24, 2025 am 12:16 AM

使用数据库存储会话的主要优势包括持久性、可扩展性和安全性。1.持久性:即使服务器重启,会话数据也能保持不变。2.可扩展性:适用于分布式系统,确保会话数据在多服务器间同步。3.安全性:数据库提供加密存储,保护敏感信息。

您如何在PHP中实现自定义会话处理?您如何在PHP中实现自定义会话处理?Apr 24, 2025 am 12:16 AM

在PHP中实现自定义会话处理可以通过实现SessionHandlerInterface接口来完成。具体步骤包括:1)创建实现SessionHandlerInterface的类,如CustomSessionHandler;2)重写接口中的方法(如open,close,read,write,destroy,gc)来定义会话数据的生命周期和存储方式;3)在PHP脚本中注册自定义会话处理器并启动会话。这样可以将数据存储在MySQL、Redis等介质中,提升性能、安全性和可扩展性。

什么是会话ID?什么是会话ID?Apr 24, 2025 am 12:13 AM

SessionID是网络应用程序中用来跟踪用户会话状态的机制。1.它是一个随机生成的字符串,用于在用户与服务器之间的多次交互中保持用户的身份信息。2.服务器生成并通过cookie或URL参数发送给客户端,帮助在用户的多次请求中识别和关联这些请求。3.生成通常使用随机算法保证唯一性和不可预测性。4.在实际开发中,可以使用内存数据库如Redis来存储session数据,提升性能和安全性。

您如何在无状态环境(例如API)中处理会议?您如何在无状态环境(例如API)中处理会议?Apr 24, 2025 am 12:12 AM

在无状态环境如API中管理会话可以通过使用JWT或cookies来实现。1.JWT适合无状态和可扩展性,但大数据时体积大。2.Cookies更传统且易实现,但需谨慎配置以确保安全性。

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

mPDF

mPDF

mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

适用于 Eclipse 的 SAP NetWeaver 服务器适配器

适用于 Eclipse 的 SAP NetWeaver 服务器适配器

将Eclipse与SAP NetWeaver应用服务器集成。

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

EditPlus 中文破解版

EditPlus 中文破解版

体积小,语法高亮,不支持代码提示功能