php调用听书插件怎样缓存音频文件_php缓存听书音频文件技巧【优化】

PHP 缓存音频需先校验本地文件是否存在且未过期,再下载并写入;用 readfile() 流式输出避免内存溢出;元数据可用 APCu 缓存,但音频本体必须存文件系统;Web 服务器应直接服务静态音频文件。

php调用听书插件怎样缓存音频文件_php缓存听书音频文件技巧【优化】

PHP 本身不直接“调用听书插件”,也没有内置的“听书插件”概念——所谓“听书”功能,本质是后端通过 PHP 下载、代理或生成音频 URL,前端(如微信小程序、H5 页面)播放。缓存音频文件的关键不在插件,而在你如何设计 PHP 的文件获取 + 存储 + 响应逻辑。

file_get_contents + file_put_contents 下载并缓存远程音频

这是最常见也最容易出问题的方式:每次请求都重新下载,没加锁、没校验、没过期控制,容易重复拉取大文件或导致并发写入损坏。

  • 务必先检查本地缓存是否存在且未过期(比如用 filemtime() 判断是否超过 24 小时)
  • 下载前用 is_writable() 确认缓存目录可写,否则静默失败
  • 下载时用 stream_context_create() 设置超时('timeout' => 30),避免卡死
  • 写入后用 clearstatcache() 避免后续 file_exists() 返回旧缓存结果
if (file_exists($cache_path) && time() - filemtime($cache_path) < 86400) {
    $audio_content = file_get_contents($cache_path);
} else {
    $context = stream_context_create(['http' => ['timeout' => 30]]);
    $audio_content = file_get_contents($remote_url, false, $context);
    if ($audio_content !== false) {
        file_put_contents($cache_path, $audio_content);
        clearstatcache(true, $cache_path);
    }
}

readfile() 流式输出缓存文件,避免内存溢出

音频文件动辄几 MB~几十 MB,若用 echo file_get_contents($path),PHP 进程会把整个文件读进内存再吐出去,极易触发 Allowed memory size exhausted 错误。

  • 必须用 readfile() 直接输出文件内容到响应体
  • 输出前手动设置正确的 Content-Type(如 audio/mpeg 对应 MP3)和 Content-Length(用 filesize()
  • 加上 ob_end_clean()flush() 避免被输出缓冲干扰
  • 不要在输出音频前有任何 echovar_dump 或空格输出,否则 HTTP 头会发送失败
if (file_exists($cache_path)) {
    ob_end_clean();
    header('Content-Type: audio/mpeg');
    header('Content-Length: ' . filesize($cache_path));
    header('Accept-Ranges: bytes');
    readfile($cache_path);
    exit;
}

apcu_cache 缓存小音频元数据,别缓存音频本体

APCu / Redis 等内存缓存适合存 URL 映射、有效期、文件大小等轻量信息,**不适合缓存几 MB 的音频二进制流**——会迅速吃光内存,还可能触发 APCu 的单值大小限制(默认约 1MB)。

LangChain

LangChain

一个开源框架,用于构建基于大型语言模型(LLM)的应用程序。

下载

立即学习PHP免费学习笔记(深入)”;

  • 只缓存 $audio_info = ['url' => $remote_url, 'expires' => time() + 86400]
  • apcu_store('audio_meta_' . $book_id, $audio_info, 86400)
  • 真正读取音频时仍走文件系统,内存缓存只做“路由”和“状态”判断
  • 如果必须用 Redis,也只存路径(如 /cache/123.mp3),而非文件内容

注意 Nginx/Apache 的静态文件缓存配置,别让 PHP 白忙

即使 PHP 正确生成了缓存文件,如果 Web 服务器没配好,浏览器每次仍会发请求到 PHP 脚本,而不是直接读取磁盘上的 MP3 文件。

  • 把音频缓存目录放在 Web 可访问路径下(如 /public/audio_cache/
  • Nginx 中加 location 规则,对 .mp3.m4a 后缀直接返回静态文件,并设置 expires 7d
  • 确保 PHP 脚本生成的文件权限为 644,Web 用户(如 www-data)有读取权
  • 避免用 index.php?audio=123 这种方式暴露 PHP 入口——直接返回 /audio_cache/123.mp3 让 Web 服务器接管更高效

真正难的不是“怎么缓存”,而是怎么让缓存路径可预测、可清理、可监控;以及当多个用户同时请求同一本听书时,如何避免 10 个进程同时去下载同一个远程音频。这些细节没处理好,缓存反而会变成性能瓶颈和故障源。

https://www.php.cn/faq/1984328.html

发表回复

Your email address will not be published. Required fields are marked *