PHP无法实现后台音频播放,仅负责通过支持HTTP Range请求的流式接口提供音频资源;真正决定后台播放的是前端JavaScript对Web Audio API或audio标签的控制及浏览器策略适配。

PHP 本身无法直接实现后台播放音频
PHP 是服务端脚本语言,执行完就结束,不持有浏览器上下文,也不可能控制音频在用户切换标签或锁屏后继续播放。所谓“PHP 调用听书插件实现后台播放”,本质是前端 JavaScript 驱动音频播放 + 后端(PHP)提供音频资源接口,PHP 只负责吐出文件路径、签名 URL、章节列表等数据。
真正起作用的是 Web Audio API 或 标签的持久化能力
能否后台播放,取决于前端是否满足浏览器的自动播放策略和后台活跃条件。关键点包括:
- 必须由用户手势(如点击按钮)触发
play(),否则多数浏览器会静音或拒绝播放 - 使用
提前加载,但不保证后台持续解码 - Android Chrome / iOS Safari 对后台音频限制极严:iOS 仅允许通过
AudioContext播放短提示音,长音频需借助「后台音频播放白名单」机制(如用户手动开启「页面音频」权限) - 部分安卓设备需启用
navigator.mediaSession并设置元信息,才能在锁屏界面显示控制栏
play.php 的正确写法:流式输出 + 正确 Header
PHP 脚本不能简单 readfile() 返回 MP3,否则无法拖拽、暂停续播、后台缓冲都会失败。必须支持 HTTP Range 请求,返回分段音频流。
- 检查
$_SERVER['HTTP_RANGE'],解析字节范围(如bytes=1024-2047) - 设置
Content-Type: audio/mpeg、Accept-Ranges: bytes、Content-Range(分段时)、Content-Length(完整或分段长度) - 用
fopen()+fseek()+fread()流式读取,避免内存溢出 - 禁止输出任何额外空格、BOM 或
echo,否则破坏二进制流
$size = filesize($filepath); $fp = fopen($filepath, 'rb');// 支持 Range 请求 if (isset($_SERVER['HTTP_RANGE'])) { list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2); list($start, $end) = array_map('intval', explode('-', $range)); $length = $end - $start + 1;
header('HTTP/1.1 206 Partial Content'); header("Content-Range: bytes $start-$end/$size"); header("Content-Length: $length"); fseek($fp, $start);} else { header('HTTP/1.1 200 OK'); header("Content-Length: $size"); }
header('Content-Type: audio/mpeg'); header('Accept-Ranges: bytes'); header('Cache-Control: public, max-age=31536000'); ob_end_clean(); fpassthru($fp); fclose($fp); ?>
后台保活靠前端,PHP 只管「别断供」
用户切走标签后,播放器是否继续响,和 PHP 几乎无关。但 PHP 接口若设计不当,会直接导致播放中断:
立即学习“PHP免费学习笔记(深入)”;
- 不要用 session 或登录态强校验每次音频请求——后台播放时 cookie 可能被限制发送
- 推荐用一次性 token(如
?token=sha256(chapter_id+salt+time)),有效期 10 分钟,避免重放攻击又不过度阻断 - 音频文件路径不要暴露真实服务器路径,用 PHP 中转,便于后续加 DRM、限速、统计
- 如果用 CDN,确保 CDN 支持 Range 请求并透传
Accept-Ranges头
真正在后台「撑住」的是前端:监听 visibilitychange、用 backgroundAudioManager(微信小程序)、或依赖系统级媒体会话(navigator.mediaSession)。PHP 做到不拖后腿,就已经完成它的使命了。











