FFmpeg 基本API av_seek_frame函数内部调用流程分析

1、av_seek_frame 函数定义

av_seek_frame() 是 FFmpeg 中用于跳转媒体文件中的某个时间点的 API,常用于播放器实现"快进/快退/定位"等功能。这个函数的作用是:根据时间戳,将解复用器(demuxer)跳转到合适的位置,以便从该位置开始读取数据帧(packet)。

可以跳到:

  • 指定时间戳的帧(通常为关键帧)
  • 或者粗略估计字节位置(用于某些格式如 MPEG-TS)
cpp 复制代码
//以下为 av_seek_frame() 的内部调用流程(位于 libavformat/utils.c)
int av_seek_frame(AVFormatContext *s, int stream_index, int64_t timestamp, int flags);
    // 调用 avformat_seek_file() 做具体的 seek 工作
    return avformat_seek_file(s, stream_index, INT64_MIN, timestamp, INT64_MAX, flags);
}

//然后:
int avformat_seek_file(...) {
    -> seek_frame_internal()
        -> ff_seek_frame_binary() 或 av_index_search_timestamp()
        -> 如果找不到直接偏移点,则调用 demuxer 的 read_seek / read_seek2
}
  • AVFormatContext *s:当前打开的媒体上下文
  • int stream_index:要 seek 的流索引(如视频流索引)。-1 表示使用全局时间戳
  • int64_t timestamp:要 seek 到的时间戳,单位是流的 time_base(或 AV_TIME_BASE 如果 stream_index == -1)
  • int flags:控制 seek 行为的标志位(详见下)
    • AVSEEK_FLAG_BACKWARD : 往后找不到就往前找(向后 seek 时的安全保障)
    • AVSEEK_FLAG_BYTE : 用字节位置 seek(不基于时间戳)
    • AVSEEK_FLAG_ANY : 允许 seek 到非关键帧(否则只能 seek 到关键帧)
    • AVSEEK_FLAG_FRAME : 使用帧编号作为 timestamp(不常用)
  • 返回值:0:成功, < 0:失败(格式不支持 seek、找不到目标、参数错等)

2、av_seek_frame 内部流程

2.1 av_seek_frame() 直接调用 avformat_seek_file()

兼容性包装,直接转调。

2.2 seek_frame_internal()

先尝试使用索引(AVStream->index_entries)做时间戳二分查找。如果不能用 index seek,会调用 demuxer 的 read_seek() / read_seek2() 回调

2.3 ff_seek_frame_binary()

通过 AVStream->index_entries(解复用器生成的 seek index,比如 .ts、.mp4 中的 moov atom 或 PTS 索引)进行二分查找。到合适关键帧的 offset

2.4 fallback

AVInputFormat->read_seek() / read_seek2()每种格式实现自己的跳转策略(比如 MPEG-TS 中的 PAT/PMT 结构)

3、av_seek_frame 调用流程图

相关推荐
uxiang_blog4 小时前
C++进阶:重载类型转换
linux·开发语言·c++
moringlightyn4 小时前
c++11可变模版参数 emplace接口 新的类功能 lambda 包装器
开发语言·c++·笔记·其他·c++11·lambda·包装器
郝学胜-神的一滴4 小时前
使用Linux系统函数递归遍历指定目录
linux·运维·服务器·开发语言·c++·软件工程
会开花的二叉树5 小时前
C++微服务 UserServer 设计与实现
开发语言·c++·微服务
我星期八休息6 小时前
C++智能指针全面解析:原理、使用场景与最佳实践
java·大数据·开发语言·jvm·c++·人工智能·python
·白小白6 小时前
力扣(LeetCode) ——11.盛水最多的容器(C++)
c++·算法·leetcode
深思慎考7 小时前
RabbitMQ 入门:基于 AMQP-CPP 的 C++ 实践指南与二次封装
开发语言·c++·分布式·rabbitmq·api
深思慎考7 小时前
Ubuntu 系统 RabbitMQ 安装指南与使用(含 C++ 客户端与 SSL 错误解决)
c++·ubuntu·rabbitmq·github·rabbitmqpp
音视频牛哥8 小时前
从“小而美”到“大而强”:音视频直播SDK的技术进化逻辑
机器学习·计算机视觉·音视频·大牛直播sdk·人工智能+·rtsp播放器rtmp播放器·rtmp同屏推流