c++怎么调用Windows摄像头接口_c++ Media Foundation框架视频捕获【方法】
技术百科
尼克
发布时间:2025-12-30
浏览: 次 Media Foundation 初始化需先调用CoInitializeEx(nullptr, COINIT_MULTITHREADED),再调用MFStartup(MF_VERSION);设备枚举须设置MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP属性;帧数据访问前必须Lock缓冲区;OnReadSample中不可直接操作UI,应PostMessage到主线程处理。
Media Foundation 初始化失败:CoInitializeEx 和 MFStartup 必须配对调用
直接调用 MFStartup 而不先初始化 COM,会导致返回 MF_E_PLATFORM_NOT_INITIALIZED。Windows 要求 Media Foundation 建立在多线程 COM 模型上,且必须显式指定 COINIT_MULTITHREADED。
-
CoInitializeEx(nullptr, COINIT_MULTITHREADED)是前提,不能省略或用CoInitialize -
MFStartup(MF_VERSION)必须在CoInitializeEx成功后调用,版本号建议用MF_VERSION(即 0x00010000) - 对应地,退出前要按顺序调用
MFShutdown()和CoUninitialize(),否则后续重复初始化可能失败
找不到可用视频采集设备:EnumDeviceSources 返回空列表
常见原因是未正确设置 MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE 属性,或设备本身被其他进程(如 Zoom、OBS、系统相机 App)独占占用。
- 枚举前需创建属性集合:
MFCreateAttributes(&pAttributes, 1),再调用pAttributes->SetGUID(MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE, MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP) - 调用
MFEnumDeviceSources(pAttributes, &ppDevices, &cDevices)后,务必检查cDevices是否 > 0;为零时不要跳过错误处理 - 若设备被占用,
ActivateObject()会返回MF_E_DEVICE_IN_USE,此时需提示用户关闭其他摄像头程序
捕获帧数据为空:IMFSample 中没有有效 IMFMediaBuffer
即使回调触发,IMFSample::GetBufferByIndex(0, &pBuffer) 可能返回 nullptr 或缓冲区长度为 0,本质是未正确锁定缓冲区或格式不匹配。
- 必须调用
pBuffer->Lock(&pData, &cbMaxLength, &cbCurrentLength)才能访问原始像素数据,仅获取指针不等于数据就绪 - 默认
捕获格式常为 MFVideoFormat_RGB32或MFVideoFormat_NV12,需通过IMFSourceReader::GetCurrentMediaType()确认实际格式,避免按错 stride 解析 - RGB32 的每行字节长(stride)未必等于
width * 4,应取MF_MT_DEFAULT_STRIDE属性值,否则图像会出现横向错位或绿条
回调线程中访问 UI 控件崩溃:IMFSourceReaderCallback 不在主线程执行
Media Foundation 默认在内部线程池中调用 OnReadSample,直接操作 Win32 窗口句柄(如 SendMessage)或 MFC/Qt 控件会引发 GDI 冲突或断言失败。
- 不要在
OnReadSample中直接调用InvalidateRect或更新CStatic图片控件 - 推荐方案:用
PostMessage向主线程窗口发送自定义消息(如WM_USER + 100),附带IMFSample*指针(需 AddRef)并在主线程中 Release - 更稳妥做法是将帧数据 memcpy 到预分配的线程安全缓冲区(如
std::atomic+std::array),再由定时器或空闲循环读取渲染
HRESULT OnReadSample(HRESULT hrStatus, DWORD dwStreamIndex, DWORD dwStreamFlags,
LONGLONG llTimestamp, IMFSample* pSample) override {
if (SUCCEEDED(hrStatus) && pSample && !(dwStreamFlags & MF_SOURCE_READER_FLAG_ENDOFSTREAM)) {
IMFMediaBuffer* pBuffer = nullptr;
BYTE* pData = nullptr;
DWORD cbMaxLength = 0, cbCurrentLength = 0;
if (SUCCEEDED(pSample->ConvertToContiguousBuffer(&pBuffer)) &&
SUCCEEDED(pBuffer->Lock(&pData, &cbMaxLength, &cbCurrentLength))) {
// ✅ 此处可安全 memcpy 数据
// ⚠️ 不要在此处 CreateDIBSection / SetDIBitsToDevice
PostMessage(hWndMain, WM_USER_FRAME_READY, 0, (LPARAM)pSample);
pSample->AddRef(); // 主线程负责 Release
}
if (pBuffer) pBuffer->Unlock();
SafeRelease(&pBuffer);
}
return S_OK;}
MF 的设备发现和帧流转依赖多个 COM 对象生命周期管理,漏掉任意一个 Release() 或提前释放 IMFSourceReader,都可能导致下一次捕获卡在等待状态。真正稳定运行的关键不在“怎么拿到帧”,而在“谁在什么时候释放了什么”。
# ai
# 多个
# 而在
# 找不到
# windows
# 什么时候
# app
# win
# word
# ui
# 循环
# 对象
# c++
# 字节
# 指针
# stream
# 接口
# 线程
# 回调
# 多线程
# usb
# zoom
# 数据访问
# 句柄
# 主线程
# Array
# 不要在
# 直接调用
# qt
# 帧数
# Foundation
# 零时
# mfc
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- 如何开启Windows的远程服务器管理工具(RSA
- Win10任务栏天气和资讯怎么关闭 Win10禁用
- Win10怎么限制单程序CPU占用上限_Win10
- Win11文件扩展名怎么显示 Win11查看文件后
- Mac如何使用听写功能_Mac语音输入打字【效率技
- 如何使用Golang实现跨域请求支持_Golang
- c# Task.ConfigureAwait(tr
- PythonWeb前后端整合项目教程_FastAP
- Windows10系统怎么查看IP地址_Win10
- Python高性能计算项目教程_NumPyCyth
- php条件判断怎么写_ifelse和switchc
- PHP主流架构怎么部署到Docker_容器化流程【
- Win11怎么设置开机自动连接宽带_Windows
- Windows如何使用BitLocker To G
- Python迭代器生成器进阶教程_节省内存与懒加载
- c++如何用AFL++进行模糊测试 c++ Fuz
- Windows10如何更改桌面图标间距_Win10
- Win11怎么退出微软账户_切换Win11为本地账
- 微信里的php文件怎么变mp4_微信接收php转m
- 如何在JavaScript中动态拼接PHP的bas
- Windows如何拦截腾讯视频广告_Windows
- Python网络日志追踪_请求定位解析【教程】
- php8.4如何实现队列任务_php8.4redi
- 如何使用Golang log设置日志输出格式_Go
- Win11如何设置开机问候语 Win11修改登录界
- Win11怎么调整屏幕亮度_Windows 11调
- Windows系统文件被保护机制阻止怎么办_权限不
- php打包exe后无法读取环境变量_变量配置方法【
- php怎么捕获异常_trycatch结构处理运行时
- win11 OneDrive怎么彻底关闭 Win1
- Windows10电脑怎么设置电源按钮_Win10
- Python与MongoDB NoSQL开发实战_
- 如何在Golang中使用time处理时间_Gola
- 使用类变量定义字符串常量时的类型安全最佳实践
- Win10系统映像怎么恢复 Win10使用系统映像
- c++如何实现多态性_c++ 虚函数表原理与动态绑
- 如何在Golang中实现自定义Benchmark_
- Win10怎么更改用户名 Win10修改账户名称操
- Windows10蓝屏代码DPC_WATCHDOG
- Win11资源管理器卡顿怎么办 Win11文件资源
- Win11怎么设置ipv4地址_Windows 1
- Win10如何关闭安全中心所有通知 Win10禁用
- 如何在Golang中实现文件下载_Golang文件
- php485读数据时阻塞怎么办_php485非阻塞
- Win10文件历史记录怎么用 Win10开启自动备
- Drupal 中 HTML 链接被双重转义导致渲染
- Windows10电脑怎么设置虚拟光驱_Win10
- Linux如何安装Golang环境_Linux下G
- 如何优化Golang程序CPU性能_Golang
- 如何使用Golang捕获测试日志_Golang t

捕获格式常为
QQ客服