当前位置: 首页 > news >正文

音视频的Buffer处理

最近在做安卓下UVC的一个案子。正好之前搞过ST方案的开机广告,这个也是我少数最后没搞成功的项目。当时也有点客观原因,当时ST要退出机顶盒市场,所以一切的支持都停了,当时啃他家播放器几十万行的代码,而且几乎没有文档,真的是非常痛苦。后面虽然功能是搞出来了,但是不稳定,持续几次后就会crash。

还记得当时最后到底层ST是用的滑动窗口缓存,双指针,一个写指针和一个读指针,当时我做了一个管道往缓存中注数据。估计还是没有完全吃透,某些细节处理有问题。正好现在又做到类似项目,所以简单总结总结相关要点。主要就是共享内存,滑动窗口,双缓冲,环形缓冲这些内容。

下面是一个简单的具有读写指针的循环缓冲区。

#include <iostream>
#include <vector>
#include <stdexcept>template<typename T>
class MediaQueue {
public:explicit MediaQueue(size_t size): buffer(size), readPtr(0), writePtr(0), count(0), maxSize(size) {}// 添加一个元素到队列中void enqueue(const T& item) {if (isFull()) {throw std::overflow_error("Queue is full");}buffer[writePtr] = item;writePtr = (writePtr + 1) % maxSize;++count;}// 从队列中读取一个元素T dequeue() {if (isEmpty()) {throw std::underflow_error("Queue is empty");}T item = buffer[readPtr];readPtr = (readPtr + 1) % maxSize;--count;return item;}// 检查队列是否为空bool isEmpty() const {return count == 0;}// 检查队列是否已满bool isFull() const {return count == maxSize;}// 获取队列中的元素数量size_t size() const {return count;}// 获取队列的最大容量size_t capacity() const {return maxSize;}private:std::vector<T> buffer;size_t readPtr;size_t writePtr;size_t count;size_t maxSize;
};

还有一种叫做乒乓buffer

就是两个buffer,一个读一个写,写完之后交换。

#include <iostream>
#include <vector>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <cstring> // For memcpy
#include <chrono> // For sleepclass PingPongBuffer {
public:PingPongBuffer(size_t bufferSize): bufferSize(bufferSize), readBufferIndex(0), writeBufferIndex(1), buffers(2, std::vector<char>(bufferSize)) {}// 写入数据到当前写缓冲区void write(const char* data, size_t size) {std::unique_lock<std::mutex> lock(mutex);while (writeBufferFull) {condVar.wait(lock);}if (size > bufferSize) {throw std::overflow_error("Data size exceeds buffer capacity");}std::memcpy(buffers[writeBufferIndex].data(), data, size);writeBufferFull = true;readBufferEmpty = false;swapBuffers();condVar.notify_all();}// 从当前读缓冲区读取数据void read(char* data, size_t size) {std::unique_lock<std::mutex> lock(mutex);while (readBufferEmpty) {condVar.wait(lock);}if (size > bufferSize) {throw std::underflow_error("Data size exceeds buffer capacity");}std::memcpy(data, buffers[readBufferIndex].data(), size);readBufferEmpty = true;writeBufferFull = false;condVar.notify_all();}private:void swapBuffers() {std::swap(readBufferIndex, writeBufferIndex);}size_t bufferSize;int readBufferIndex;int writeBufferIndex;std::vector<std::vector<char>> buffers;bool readBufferEmpty = true;bool writeBufferFull = false;std::mutex mutex;std::condition_variable condVar;
};void producer(PingPongBuffer& buffer) {const char* messages[] = {"Message 1", "Message 2", "Message 3"};for (const char* message : messages) {std::this_thread::sleep_for(std::chrono::seconds(1)); // Simulate workbuffer.write(message, std::strlen(message) + 1);std::cout << "Produced: " << message << std::endl;}
}void consumer(PingPongBuffer& buffer) {char data[1024];for (int i = 0; i < 3; ++i) {buffer.read(data, 1024);std::cout << "Consumed: " << data << std::endl;}
}int main() {size_t bufferSize = 1024;PingPongBuffer buffer(bufferSize);std::thread producerThread(producer, std::ref(buffer));std::thread consumerThread(consumer, std::ref(buffer));producerThread.join();consumerThread.join();return 0;
}

空了有时间看看V4L2和ffmpeg这方面的内容再更新一下吧。。。

相关文章:

  • Parallelize your massive SHAP computations with MLlib and PySpark
  • NTFS和exFAT哪个性能好 U盘格式化NTFS好还是exFAT好 mac不能读取移动硬盘怎么解决
  • 深信服科技:2023网络安全深度洞察及2024年趋势研判报告
  • [实践篇]13.29 再来聊下Pass Through设备透传
  • sixLabors.ImageSharp图片截取
  • 中介子方程二十九
  • win10远程桌面连接端口,远Win10远程桌面连接端口修改及无法连接解决方案
  • Vue67-Vuex简介
  • Leetcode3185. 构成整天的下标对数目 II
  • 基于IDEA的Maven(依赖介绍和引用)
  • idea怎么重复启动实例
  • Neo4j 创建关系
  • 代码随想录算法训练营第31天| 134. 加油站、135. 分发糖果、860.柠檬水找零、 406.根据身高重建队列
  • C#事件详解及应用示例
  • 码住!详解时序数据库不同分类与性能对比
  • 《微软的软件测试之道》成书始末、出版宣告、补充致谢名单及相关信息
  • android图片蒙层
  • canvas 高仿 Apple Watch 表盘
  • HTML5新特性总结
  • idea + plantuml 画流程图
  • SegmentFault 社区上线小程序开发频道,助力小程序开发者生态
  • SpriteKit 技巧之添加背景图片
  • 百度贴吧爬虫node+vue baidu_tieba_crawler
  • 高程读书笔记 第六章 面向对象程序设计
  • 关于extract.autodesk.io的一些说明
  • 算法-插入排序
  • 优秀架构师必须掌握的架构思维
  • 【运维趟坑回忆录 开篇】初入初创, 一脸懵
  • 微龛半导体获数千万Pre-A轮融资,投资方为国中创投 ...
  • ​​​​​​​ubuntu16.04 fastreid训练过程
  • ​iOS实时查看App运行日志
  • ​Linux Ubuntu环境下使用docker构建spark运行环境(超级详细)
  • #### go map 底层结构 ####
  • #NOIP 2014# day.1 T3 飞扬的小鸟 bird
  • #php的pecl工具#
  • $(document).ready(function(){}), $().ready(function(){})和$(function(){})三者区别
  • (2)空速传感器
  • (rabbitmq的高级特性)消息可靠性
  • (Repost) Getting Genode with TrustZone on the i.MX
  • (Ruby)Ubuntu12.04安装Rails环境
  • (规划)24届春招和25届暑假实习路线准备规划
  • (原创)Stanford Machine Learning (by Andrew NG) --- (week 9) Anomaly DetectionRecommender Systems...
  • (转)chrome浏览器收藏夹(书签)的导出与导入
  • (转)Linux NTP配置详解 (Network Time Protocol)
  • .bat批处理(一):@echo off
  • .NET 4 并行(多核)“.NET研究”编程系列之二 从Task开始
  • .Net 4.0并行库实用性演练
  • .NET/C# 推荐一个我设计的缓存类型(适合缓存反射等耗性能的操作,附用法)
  • @DataRedisTest测试redis从未如此丝滑
  • @transaction 提交事务_【读源码】剖析TCCTransaction事务提交实现细节
  • [ vulhub漏洞复现篇 ] ECShop 2.x / 3.x SQL注入/远程执行代码漏洞 xianzhi-2017-02-82239600
  • [20170728]oracle保留字.txt
  • [BUUCTF NewStarCTF 2023 公开赛道] week3 crypto/pwn
  • [C/C++]数据结构 栈和队列()
  • [caffe(二)]Python加载训练caffe模型并进行测试1