php output_buffering 缓存使用

buffer是一个内存地址空间,linux系统默认大小一般为4096(4kb)。主要用于存储速度不同步的设备或优先级不同的设备之间传递数据的区域。

通过buffer可以使进程之间的相互等待变少。

例如,当你打开一个文本编辑器,输入一个字符时,系统并不会立即写入磁盘。而是存储在buffer中,当写满一个buffer时,才会把buffer的内容写入磁盘。

当然也可以使用flush强制将buffer的数据写入磁盘。

在php中,例如 echo,print,输出并不会立即通过tcp传递给浏览器输出。而是将数据写入php buffer。当一个php buffer写满,才会通过tcp传递给浏览器。

但是浏览器也是有缓存的,一些版本的浏览当数据达到256字节的时候才输出内容,flush()可以将等待输出的内容立即发送到客户端,而ob_flush()只有等待到buffer满的时候才输出。

echo /print -> php output_buffer -> tcp buffer -> browser


php output_buffering

默认情况下,php buffer是开启的,默认值是4096(4kb)。可以在php.ini中找到output_buffering 配置。知道buffer写满,才会发送到浏览器。

也可以设置ob_start(),这样即使超过默认值(4kb)也不会立即发送到浏览器,只到脚本运行完成,或者调用了ob_end_flush方法,才会输出到浏览器。


1.output_buffering=4096,输出较少的数据(少于一个buffer)

    <?php  
    for($i=0; $i<10; $i++){  
        echo $i.'<br>';  
        sleep(2);  
    }  
    ?>  

运行结果:等所有脚本全部运行完成后,才输出,因为数据未满一个buffer的大小。


2.output_buffering=4096,输出较少的数据(少于一个buffer),关闭output_buffering,修改php.ini的output_buffering=0

    <?php  
    for($i=0; $i<10; $i++){  
        echo $i.str_repeat(" ",500).'<br>'; // 当接受到的256(甚至更多,每个浏览器不一样)个字节以后才开始显示该页面,所以必须发送一些额外的空格来让这些浏览器显示页面内容。  
        flush();  
        sleep(1);  
    }  
    ?>  


运行结果:因为禁用了php buffering,不需要等到脚本运行完毕就可以输出,数据没有在php buffer停留,可以看到断断续续间歇性输出。echo -> tcp buffer ->browser


3.当output_buffering=4096,输出较大数据(大于一个buffer),不使用ob_start()

[php] view plain copy
  1.     <?php  
        for($i=0; $i<10; $i++){  
            echo file_get_contents('f.txt').$i.'<br>';  
            sleep(1);  
        }  
        ?>  


运行结果:f.txt为一个大于4kb的文件,因为大于buffer默认值,buffer空间不够用,每当满一个buffer就会输出,所以可以看到间歇性输出。


4.当output_buffering=4096,输出较大数据(大于一个buffer),使用ob_start()

    <?php  
    ob_start();  
    for($i=0; $i<10; $i++){  
        echo file_get_contents('f.txt').$i.'<br>';  
        sleep(1);  
    }  
    ?>  


运行结果:因为使用了ob_start(),会为buffer设置足够大的空间,因此会保存到脚本执行完毕后才会输出。


output_buffering 方法

1.ob_start
激活output_buffering机制,一旦激活,脚本不再直接输出到浏览器,而是暂时写入php buffering区域。直到脚本运行完毕后,才发送。

2.ob_get_contents
获取php buffering中的数据,注意:要在ob_end_clean()前调用,否则只会得到空字符。

3.ob_end_flush 和 ob_end_clean
ob_end_flush 会输出php buffering 中的数据,但不会清空。
ob_end_clean 不会输出,只会清空php buffering中的数据。 


注意:
ob_flush/flush在手册中的描述, 都是刷新输出缓冲区, 并且还需要配套使用, 所以会导致很多人迷惑…
其实, 他们俩的操作对象不同, 有些情况下, flush根本不做什么事情..
ob_*系列函数, 是操作PHP本身的输出缓冲区.
所以, ob_flush是刷新PHP自身的缓冲区.
而flush, 严格来讲, 这个只有在PHP做为apache的Module(handler或者filter)安装的时候, 才有实际作用. 它是刷新WebServer(可以认为特指apache)的缓冲区.
在apache module的sapi下, flush会通过调用sapi_module的flush成员函数指针, 间接的调用apache的api: ap_rflush刷新apache的输出缓冲区, 当然手册中也说了, 有一些apache的其他模块, 可能会改变这个动作的结果..
有些Apache的模块,比如mod_gzip,可能自己进行输出缓存,这将导致flush()函数产生的结果不会立即被发送到客户端浏览器。
甚至浏览器也会在显示之前,缓存接收到的内容。例如 Netscape浏览器会在接受到换行或 html 标记的开头之前缓存内容,并且在接受到 </table> 标记之前,不会显示出整个表格。
一些版本的 Microsoft Internet Explorer 只有当接受到的256个字节以后才开始显示该页面,所以必须发送一些额外的空格来让这些浏览器显示页面内容。

所以, 正确使用俩者的顺序是。 先ob_flush, 然后flush。


当然, 在其他sapi下,,不调用flush也可以, 只不过为了保证你代码的可移植性, 建议配套使用。

你可能感兴趣的:(php output_buffering 缓存使用)