php 中的 output_buffering 缓存区

发布于 2021-07-27 12:38:39 字数 3540 浏览 1407 评论 0

php 默认是打开 output_buffering 可以在 php.ini 文件中找到。当执行 echo、print 的时候,输出并没有立即通过tcp传给客户端浏览器显示,而是将数据写入 php buffer。php output_buffering 机制,意味在 tcp buffer 之前,建立了一新的队列,数据必须经过该队列。

当一个 php buffer 写满的时候,脚本进程会将 php buffer 中的输出数据交给系统内核交由 tcp 传给浏览器显示。所以,数据会依次写到这几个地方 echo/print -> php buffer -> tcp buffer -> browser buffer

当 echo、print 等输出用户数据的时候,输出数据都会写入到 php output_buffering 中,直到 output_buffering 写满,会将这些数据通过tcp传送给浏览器显示。你也可以通过 ob_start()手动激活php output_buffering机制,使得即便输出超过了1kb数据,也不真的把数据交给tcp传给浏览器,因为 ob_start() 将 php buffer 空间设置到了足够大 。只有直到脚本结束,或者调用 ob_end_flush 函数,才会把数据发送给客户端浏览器。

output_buffering = 4096

一、什么时候使用 ob_flush?

ob_系列函数,是操作PHP本身的输出缓存区。所以,ob_flush 是刷新 PHP 自身的 *缓存区。

ob_flush() : 这个函数将送出缓存区的内容(如果里边有内容的话)。如果想进一步处理缓存区中的内容,必须在 ob_flush() 之前调用 ob_get_contents() ,因为在调用 ob_flush() 之后缓存区内容将被丢弃。

二、什么时候使用 flush?

flush() 函数,是操作PHP本身的输出 缓冲区

flush() : 刷新PHP程序的缓冲,而不论PHP执行在何种情况下(CGI ,web服务器等等)。该函数将当前为止程序的所有输出发送到用户的浏览器。

flush(),严格来讲,这个只有在PHP做为apache的Module(handler 或者 filter)安装的时候,才有实际作用。它是刷新 WebServer(可以认为特指 apache)的缓冲区,使用 nginx 作为 WebServer 是不用生效的。

在没有开启缓存时,脚本输出的内容都在服务器端处于等待输出的状态(缓冲区) ,flush()可以将等待输出的内容立即发送到客户端。

开启缓存后,脚本输出的内容存入了输出缓存中(缓存区) ,这时没有处于等待输出状态的内容(缓冲区),你直接使用flush()不会向客户端发出任何内容。而 ob_flush()的作用就是将本来存在输出缓存中的内容取出来,设置为等待输出状态,但不会直接发送到客户端 ,这时你就需要先使用 ob_flush()再使用flush(),客户端才能立即获得脚本的输出。

三、兼容方案

那么,下面介绍一种通用的写法,代码如下:

<?php

header('X-Accel-Buffering: no');//告诉nginx不要缓存,直接输出给浏览器

ob_start();//有没有都一样


for($j = 1; $j <= 6; $j++) {
    echo $j . '****';
    echo ob_get_clean();
    flush();
    sleep(1);

还有一种做法就是修改 nginx 配,缓存关了。影响较大

gzip off;
proxy_buffering off;//关闭proxy nginx的缓存区
fastcgi_keep_conn on;

之后就可以直接使用 ob_flush();flush();

For($j = 1; $j <= 6; $j++) {
    echo $j . '*';
    ob_flush();
    flush();
    sleep(1);
}

解释下 proxy_buffering

proxy_buffering 主要是实现被代理服务器的数据和客户端的请求异步。

为了方便理解,我们定义三个角色,A为客户端,B为代理服务器,C为被代理服务器。

当proxy_buffering开启,A发起请求到B,B再到C,C反馈的数据先到B的buffer上,然后B会根据proxy_busy_buffer_size来决定什么时候开始把数据传输给A。

在此过程中,如果所有的buffer被写满,数据将会写入到temp_file中。

相反,如果proxy_buffering关闭,C反馈的数据实时地通过B传输给A。

以下配置,都是针对每一个http请求的:

1. proxy_buffering on;

该参数设置是否开启proxy的buffer功能,参数的值为on或者off。
如果这个设置为off,那么proxy_buffers和proxy_busy_buffers_size这两个指令将会失效。
但是无论proxy_buffering是否开启,proxy_buffer_size都是生效的

2. proxy_buffer_size 4k;

该参数用来设置一个特殊的buffer大小的。
从被代理服务器(C)上获取到的第一部分响应数据内容到代理服务器(B)上,通常是header,就存到了这个buffer中。
如果该参数设置太小,会出现502错误码,这是因为这部分buffer不够存储header信息。建议设置为4k。

3. proxy_buffers 8 4k;

这个参数设置存储被代理服务器上的数据所占用的buffer的个数和每个buffer的大小。
所有buffer的大小为这两个数字的乘积。

4. proxy_busy_buffer_size 16k;

在所有的buffer里,我们需要规定一部分buffer把自己存的数据传给A,这部分buffer就叫做busy_buffer。
proxy_busy_buffer_size参数用来设置处于busy状态的buffer有多大。

对于B上buffer里的数据何时传输给A,我个人的理解是这样的:
1)如果完整数据大小小于busy_buffer大小,当数据传输完成后,马上传给A;
2)如果完整数据大小不少于busy_buffer大小,则装满busy_buffer后,马上传给A;

5. proxy_temp_path

语法:proxy_temp_path path [level1 level2 level3]
定义proxy的临时文件存在目录以及目录的层级。

例:proxy_temp_path /usr/local/nginx/proxy_temp 1 2;
其中/usr/local/nginx/proxy_temp为临时文件所在目录,1表示层级1的目录名为1个数字(0-9),2表示层级2目录名为2个数字(00-99)

6. proxy_max_temp_file_size

设置临时文件的总大小,例如 proxy_max_temp_file_size 100M;

7. proxy_temp_file_wirte_size

设置同时写入临时文件的数据量的总大小。通常设置为8k或者16k。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

0 文章
0 评论
84961 人气
更多

推荐作者

已经忘了多久

文章 0 评论 0

15867725375

文章 0 评论 0

LonelySnow

文章 0 评论 0

走过海棠暮

文章 0 评论 0

轻许诺言

文章 0 评论 0

信馬由缰

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文