news 2026/4/3 20:20:55

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/3 2:41:28

至少148亿元!近三年受害企业支付勒索软件赎金金额创新高

至少148亿元!近三年受害企业支付勒索软件赎金金额创新高 据美国财政部下属机构统计,2022-2024年期间,受害企业仅通过美国金融机构,就至少向勒索软件组织支付了超148亿元赎金,创下历史新高。 安全内参12月8日报道&…

作者头像 李华
网站建设 2026/4/3 1:20:30

磁吸抗金属标签能够有效解决高温金属环境中的物联网追踪难题

lassorfid 在钢铁冶金高炉旁、汽车制造涂装线、石油化工反应釜等场景中,高温炙烤与复杂金属环境的双重挑战导致传统RFID标签频繁失效:芯片因150℃以上高温熔断、金属表面电磁波反射引发信号中断、粘贴式安装于设备检修时反复脱落……这些痛点严重制约了…

作者头像 李华
网站建设 2026/4/1 18:07:24

AutoGPT upsell 推荐文案生成器

AutoGPT:当AI开始“自己做事” 想象一下,你只需要说一句:“帮我写一份关于国内AI客服市场现状的报告”,然后去喝杯咖啡——15分钟后回来,发现不仅资料已经搜集完毕,连结构清晰、数据详实的初稿都已生成。这…

作者头像 李华
网站建设 2026/4/1 19:13:28

GPT-5.2发布效率提升11倍,成本仅为1%的AI助手来了【建议收藏】

OpenAI发布GPT-5.2模型系列,包含Instant、Thinking和Pro三个版本,专为专业知识型工作打造。该模型在多项基准测试中表现优异,超越Gemini 3和Claude Opus 4.5,在44个职业评测中70.9%的表现优于或持平人类专家,效率提升1…

作者头像 李华
网站建设 2026/4/1 8:14:55

GitHub Issue模板设计:规范PyTorch项目问题反馈

GitHub Issue模板设计:规范PyTorch项目问题反馈 在深度学习项目的协作开发中,一个常见的尴尬场景是:开发者提交了一个Bug报告,维护者兴冲冲地拉代码复现,结果第一行就报错——“torch not found”。再一问,…

作者头像 李华