php调用听书插件怎么实现后台播放_php听书插件后台播放实现法【后台】

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/mpegAccept-Ranges: bytesContent-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 接口若设计不当,会直接导致播放中断:

  • 不要用 session 或登录态强校验每次音频请求——后台播放时 cookie 可能被限制发送
  • 推荐用一次性 token(如 ?token=sha256(chapter_id+salt+time)),有效期 10 分钟,避免重放攻击又不过度阻断
  • 音频文件路径不要暴露真实服务器路径,用 PHP 中转,便于后续加 DRM、限速、统计
  • 如果用 CDN,确保 CDN 支持 Range 请求并透传 Accept-Ranges

真正在后台「撑住」的是前端:监听 visibilitychange、用 backgroundAudioManager(微信小程序)、或依赖系统级媒体会话(navigator.mediaSession)。PHP 做到不拖后腿,就已经完成它的使命了。