登录
首页 >  文章 >  php教程

PHP实时输出Chunk大小能调整吗

时间:2026-01-27 12:06:44 450浏览 收藏

目前golang学习网上已经有很多关于文章的文章了,自己在初次阅读这些文章中,也见识到了很多学习思路;那么本文《PHP实时输出Chunk大小可调整吗》,也希望能帮助到大家,如果阅读完后真的对你学习文章有帮助,欢迎动动手指,评论留言并分享~

PHP实时输出的chunk大小由Web服务器和PHP缓冲机制共同决定,无法通过PHP函数直接设置;关键需禁用zlib压缩、关闭output_buffering,并配置Nginx/Apache禁用gzip、buffering等代理层缓冲。

php实时输出chunk大小能改吗_php实时输出chunk调优【教程】

PHP 实时输出的 chunk 大小由 Web 服务器和 PHP 缓冲机制共同决定,不能直接通过 PHP 函数设置“chunk 大小”

很多人以为 ob_flush()flush() 能控制每次发给浏览器的数据块大小,其实它们只是触发刷新,真正发送的 chunk 尺寸取决于底层:Nginx 默认 4KB、Apache 的 BufferedLogsSendBufferSize、PHP 的 output_buffering 和 zlib 压缩开关。你调用一次 echo "a" + flush(),浏览器可能等 4096 字节才收到第一个 TCP 包。

禁用 zlib.output_compression 是实时输出的前提,否则所有输出被压缩缓冲截断

这是最常踩的坑:即使关了 output_buffering,只要 zlib.output_compression = On(或在脚本里开了 ob_start('ob_gzhandler')),PHP 会强制累积数据直到达到 zlib 内部缓冲阈值(通常是 4KB),flush() 完全无效。

  • 检查配置:php -i | grep zlib.output_compression,确认值为 Off
  • 运行时关闭:ini_set('zlib.output_compression', '0');(必须在任何输出前调用)
  • 如果用了 ob_start(),确保没传入压缩回调函数

Web 服务器层才是真正的 chunk 控制点,PHP 层只能“配合”

Nginx 默认启用 gzipproxy_buffering,这两者都会吞掉小 chunk;Apache 的 mod_deflateEnableSendfile 同理。PHP 没有 API 能告诉 Nginx “现在发 1 字节”,只能靠服务端配置让管道变“细”:

  • Nginx:加 gzip off;proxy_buffering off;chunked_transfer_encoding on;
  • Apache:禁用 mod_deflate,设 EnableSendfile off,并确保 SetOutputFilter none
  • PHP-FPM 场景下,还要检查 fastcgi_buffering off;(Nginx 1.11.5+)

最小可行实时输出模式:关缓冲 + 关压缩 + 小写 + 强刷

以下代码在正确服务端配置下,可做到接近逐字符推送(实际仍受 TCP/IP 栈和浏览器解析策略影响):

<?php // 必须放在第一行
if (function_exists('apache_setenv')) {
    apache_setenv('no-gzip', '1');
}
ini_set('zlib.output_compression', '0');
ini_set('output_buffering', '0');
ini_set('implicit_flush', '1');

// 清空所有可能存在的缓冲
while (ob_get_level()) ob_end_clean();

// 发送 header 防止浏览器缓存 & 声明流式响应
header('Content-Encoding: identity');
header('X-Accel-Buffering: no');
header('Cache-Control: no-cache');
header('Content-Type: text/plain; charset=utf-8');

for ($i = 0; $i < 10; $i++) {
    echo "tick {$i}\n";
    // 关键:ob_flush() 刷 PHP 输出缓冲,flush() 刷 Web 服务器缓冲
    ob_flush();
    flush();
    usleep(500000); // 半秒间隔,便于观察
}

注意:ob_flush()flush() 必须成对出现;usleep() 不是必须,但没有它,循环太快会导致多个 tick 合并在一个 TCP 包里发出——这不是 PHP 的问题,是内核 Nagle 算法在起作用。

真正难调的从来不是 PHP 代码,而是跨层协同:PHP 缓冲关了,Web 服务器没关压缩;服务端调好了,CDN 又加了一层缓冲;甚至某些浏览器(如 Safari)对非 text/event-stream 类型的流式响应会静默攒包。调优时得一层层抓包验证,别只盯着 echoflush()

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《PHP实时输出Chunk大小能调整吗》文章吧,也可关注golang学习网公众号了解相关技术文章。

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>