PHP怎样读取视频元数据信息_PHP读取视频元数据信息途径【技巧】

PHP解析视频元数据必须依赖外部工具或扩展:首选FFmpeg命令行(ffprobe),其次php-ffmpeg扩展,或轻量级getID3库;三者在兼容性、精度和部署难度上各有取舍。

php怎样读取视频元数据信息_php读取视频元数据信息途径【技巧】

PHP 本身不内置视频元数据解析能力,getimagesize() 对视频无效,exif_read_data() 仅支持 JPEG/TIFF,直接用 fopen() 读二进制头也极不可靠——必须依赖外部工具或扩展。

用 FFmpeg 命令行 + shell_exec() 最通用

这是目前最稳定、兼容性最好的方案,几乎所有 Linux/macOS 服务器都可部署,Windows 需额外配置 FFmpeg 路径。

  • 确保系统已安装 FFmpeg:运行 ffmpeg -version 能输出版本号
  • PHP 中调用:ffprobe(FFmpeg 的元数据专用工具)比 ffmpeg -i 更轻量、更结构化
  • 务必对视频路径做 escapeshellarg() 过滤,否则存在命令注入风险
  • 推荐使用 JSON 输出格式,便于 json_decode() 解析,避免正则匹配的脆弱性
function getVideoMetadata($videoPath) {
    $escapedPath = escapeshellarg($videoPath);
    $cmd = "ffprobe -v quiet -print_format json -show_format -show_streams {$escapedPath}";
    $output = shell_exec($cmd);
    return json_decode($output, true);
}

// 示例返回中重点关注:
// $meta['format']['duration'] → 总时长(秒,字符串)
// $meta['streams'][0]['width'] / ['height'] → 分辨率
// $meta['streams'][0]['codec_name'] → 编码器(如 'h264')
// $meta['streams'][0]['r_frame_rate'] → 帧率(如 '30/1',需计算)

php-ffmpeg 扩展(需编译安装)

这是纯 PHP 封装层,底层仍调用 FFmpeg 二进制,但提供了面向对象接口,适合中大型项目统一管理。

  • 必须通过 pecl install ffmpeg 或源码编译安装,不是 composer require 能解决的
  • PHP 版本兼容性敏感:PHP 8.0+ 需用 ffmpeg 4.x 分支,旧版扩展可能报 Class 'ffmpeg_movie' not found
  • 不支持 Windows 下的线程安全(TS)PHP 构建,生产环境慎用
  • 获取时长需注意:$movie->getFrameCount() / $movie->getFrameRate() 不可靠,应优先读 $movie->getDuration()

getID3 库(纯 PHP,无依赖)

适合不能装 FFmpeg、又需要快速获取基础信息(如时长、尺寸、编码)的轻量场景,但对 HEVC、AV1、QuickTime 新格式支持滞后。

磁力开创

磁力开创

快手推出的一站式AI视频生产平台

下载

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

  • 通过 Composer 安装:composer require james-heinrich/getid3
  • 它本质是“猜”格式:基于文件头和字节偏移解析,遇到非标准 MP4(如由手机直录带私有 atom)可能返回空或错误 duration
  • 关键字段命名与 FFmpeg 不同:$info['video']['resolution_x'] 而非 width$info['playtime_seconds'] 是浮点数时长
  • 不解析音频流详细参数(如 AAC 的profile),只给 codec 字符串,精度有限

真正麻烦的不是“怎么读”,而是“读到的数据是否可信”:移动端录的 MP4 常缺 moov box(放在文件开头),FFmpeg 默认能容错,而 getID3 可能直接放弃;某些剪辑软件导出的 MOV 会把帧率写成 0/0,需要 fallback 到采样估算。别跳过验证步骤。

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

发表回复

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