news 2026/4/23 0:11:01

FFmepg-- 33-ffplay源码-FrameQueue 环形缓冲区涉及以及保持画面

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
FFmepg-- 33-ffplay源码-FrameQueue 环形缓冲区涉及以及保持画面

文章目录

      • FrameQueue 使用环形缓冲区的原因
        • 背景需求
        • 环形缓冲区优势
      • FrameQueue vs 普通队列:核心差异
      • keep_last 机制详解
        • 设计目的
        • 核心字段含义
        • 关键函数:frame_queue_peek()
        • 关键函数:frame_queue_next()
      • 运行示例(视频队列,max_size=3, keep_last=1)
      • frame_queue_nb_remaining() 如何计算?
      • keep_last 总结

FrameQueue 使用环形缓冲区的原因

背景需求

视频播放中解码线程持续向队列写入帧(生产者),渲染线程从队列读取帧用于显示(消费者)。需要支持保留当前显示帧、预读下一帧(用于计算显示间隔、插帧等)、暂停时画面不消失、低延迟和高性能。普通动态队列(如链表)频繁 malloc/free 会带来性能开销和内存碎片,不适合实时音视频场景。

环形缓冲区优势

ffplay 的 FrameQueue 是一个固定大小的数组 + 读写指针结构:

typedefstructFrameQueue{AVFrame*queue[MAX_FRAME_QUEUE_SIZE];// 固定数组,提前分配intrindex;// 读指针(逻辑上指向“已读但未释放”的帧)intwindex;// 写指针intsize;// 当前有效帧数intmax_size;// 最大容量(通常 audio=9, video=3, sub=16)intkeep_last;// 是否保留最后一帧intrindex_shown;// 标记当前 rindex 帧是否已被“显示过”SDL_mutex*mutex;SDL_cond*cond;}FrameQueue;

优点总结:

  • 内存一次性分配,无碎片
  • 指针通过 % max_size 循环使用,O(1) 读写
  • 支持“只读不删”(peek)、“延迟删除”(next)

FrameQueue vs 普通队列:核心差异

特性普通队列(如 std::queue)ffplay FrameQueue
出队行为pop() 立即释放内存frame_queue_next() 才真正释放
读操作front() 后必须 pop可多次 peek() 查看当前/下一帧
最后一帧出队即销毁若 keep_last=1,即使出队也保留
用途通用数据传输音视频渲染专用(需保留历史帧)

关键区别:“读” ≠ “消费”。FrameQueue 允许“查看但不移除”,这是实现流畅渲染的基础。

keep_last 机制详解

设计目的
  • 暂停时保持画面:不能因为“已显示”就立刻释放帧
  • 计算帧间隔:需要 lastvp(上一帧)和 vp(当前帧)的 PTS 差值
  • 防止黑屏:在新帧未到达前,继续显示旧帧
核心字段含义
intrindex;// 指向“逻辑上最后一个保留的帧”(通常是 lastvp)intrindex_shown;// =0 表示 rindex 帧尚未作为“当前帧”显示过// =1 表示 rindex 帧已是“上一帧”,当前帧是 (rindex+1)%sizeintkeep_last;// =1 表示启用保留机制(video/subtitle 启用,audio 不启用)
关键函数:frame_queue_peek()
// 获取当前应显示的帧(不移动指针)staticinlineFrame*frame_queue_peek(FrameQueue*q){return&q->queue[(q->rindex+q->rindex_shown)%q->max_size];}

若 rindex_shown=0 → 返回 rindex 帧(即 lastvp,也是当前帧)
若 rindex_shown=1 → 返回 (rindex+1) 帧(当前帧),而 rindex 是 lastvp

关键函数:frame_queue_next()
staticinlinevoidframe_queue_next(FrameQueue*q){if(q->keep_last&&!q->rindex_shown){// 第一次调用 next():仅标记 rindex_shown=1,不移动 rindexq->rindex_shown=1;return;}// 真正释放 rindex 帧,并移动指针av_frame_unref(q->queue[q->rindex]);q->rindex=(q->rindex+1)%q->max_size;q->rindex_shown=0;// 新的 rindex 尚未作为“当前帧”显示q->size--;SDL_CondSignal(q->cond);}

运行示例(视频队列,max_size=3, keep_last=1)

假设解码线程写入了 3 帧:F0, F1, F2

初始状态(刚写完):

queue = [F0, F1, F2] windex = 0(循环回绕) rindex = 0 rindex_shown = 0 size = 3

第一次渲染(显示 F0):

  • frame_queue_peek() → 返回 queue[(0+0)%3] = F0
  • 显示 F0
  • 调用 frame_queue_next():
    • 因为 keep_last=1 && rindex_shown=0 → 仅设 rindex_shown=1
    • rindex 仍为 0,F0 未被释放!
      此时:
rindex = 0 (F0 保留为 lastvp) rindex_shown = 1 → 当前帧是 (0+1)=F1 size = 3(未减少!)

第二次渲染(显示 F1):

  • frame_queue_peek() → 返回 queue[(0+1)%3] = F1
  • 显示 F1
  • 调用 frame_queue_next():
    • 现在 rindex_shown=1 → 执行真实出队:
      • av_frame_unref(F0) → 释放 F0
      • rindex = (0+1)%3 = 1
      • rindex_shown = 0
      • size = 2
        此时:
queue = [__, F1, F2] (F0 已释放) rindex = 1 → 指向 F1(作为新的 lastvp) rindex_shown = 0 → 当前帧仍是 F1(下一次 peek 还是 F1)

注意:F1 被“保留”了两次——第一次作为“当前帧”显示,第二次作为“lastvp”供下次同步参考。

frame_queue_nb_remaining() 如何计算?

staticintframe_queue_nb_remaining(FrameQueue*q){returnq->size-q->rindex_shown;}
  • size:队列中总帧数(包括保留的 lastvp)
  • rindex_shown:若为 1,说明 lastvp 已“转正”为历史帧,当前帧是下一个,因此可显示的帧数 = size - 1
  • 若为 0,说明当前帧就是 rindex,所有 size 帧都可用

示例:

  • size=3, rindex_shown=1 → 可显示帧数 = 2(当前帧 + 下一帧)
  • size=1, rindex_shown=0 → 可显示帧数 = 1(只有当前帧)

这对 video_refresh 判断是否该丢帧或等待至关重要。

keep_last 总结

场景作用
暂停播放保留 lastvp,画面不黑
计算帧率lastvp.pts 与 vp.pts 做差
音视频同步视频时钟基于 lastvp 更新
低内存占用固定 3 帧缓存,避免堆积
流畅渲染支持“显示当前帧 + 预读下一帧”
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 13:37:50

博奥龙Hybridoma Feeder添加因子(含常见问题解答及客户评价)

01、什么是饲养层细胞? 在体外细胞培养中,对于一些难以生长或数量稀少的目的细胞(如杂交瘤细胞),需要辅助支持。通常的做法是预先在培养器皿底部铺上一层活细胞(如原代细胞或静息的肿瘤细胞)&a…

作者头像 李华
网站建设 2026/4/23 12:29:21

LobeChat能否集成Figma插件?设计协作新范式

LobeChat 与 Figma 插件集成:重塑设计协作的智能路径 在今天的数字产品开发流程中,设计师、产品经理和工程师之间的协作效率,往往决定了项目推进的速度与质量。一个常见的场景是:产品经理在会议中突然发问,“最新的登录…

作者头像 李华
网站建设 2026/4/5 13:23:07

Git分支管理策略:适用于大型PyTorch项目协作开发

Git分支管理策略:适用于大型PyTorch项目协作开发 在现代AI研发中,一个再常见不过的场景是:团队成员A兴奋地宣布“我的新模型准确率提升了3%”,可当其他人试图复现结果时,却遭遇了五花八门的问题——CUDA版本不兼容、依…

作者头像 李华
网站建设 2026/4/13 23:55:46

“从开题答辩到终稿提交:一位普通本科生如何借助AI科研助手,在不碰红线的前提下走通毕业论文全流程?”

在高校教学管理日益规范、学术伦理要求愈发严格的今天,本科毕业论文早已不是“随便写写就能过”的形式任务。它既是学术训练的终点,也是科研思维的起点。然而,对于首次接触系统性研究的本科生而言,这场旅程往往伴随着三重困境&…

作者头像 李华
网站建设 2026/4/18 5:37:54

Stable Diffusion AIGC 视觉设计实战教程之 06-提示词应用技巧

正向提示词技巧 正向提示词基础 在 Stable Diffusion 的图像生成过程中,正向提示词书写公式扮演着至关重要的角色。在构建 Stable Diffusion 正向提示词时,主要包含画面内容(主体、动作、道具、环境等)、画面风格、画面构图、通用…

作者头像 李华