0

0

c++怎么调用Windows摄像头接口_c++ Media Foundation框架视频捕获【方法】

尼克

尼克

发布时间:2025-12-30 09:40:03

|

128人浏览过

|

来源于php中文网

原创

Media Foundation 初始化需先调用CoInitializeEx(nullptr, COINIT_MULTITHREADED),再调用MFStartup(MF_VERSION);设备枚举须设置MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP属性;帧数据访问前必须Lock缓冲区;OnReadSample中不可直接操作UI,应PostMessage到主线程处理。

c++怎么调用windows摄像头接口_c++ media foundation框架视频捕获【方法】

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_RGB32MFVideoFormat_NV12,需通过 IMFSourceReader::GetCurrentMediaType() 确认实际格式,避免按错 stride 解析
  • RGB32 的每行字节长(stride)未必等于 width * 4,应取 MF_MT_DEFAULT_STRIDE 属性值,否则图像会出现横向错位或绿条

回调线程中访问 UI 控件崩溃:IMFSourceReaderCallback 不在主线程执行

Media Foundation 默认在内部线程池中调用 OnReadSample,直接操作 Win32 窗口句柄(如 SendMessage)或 MFC/Qt 控件会引发 GDI 冲突或断言失败。

Winston AI
Winston AI

强大的AI内容检测解决方案

下载

立即学习C++免费学习笔记(深入)”;

  • 不要在 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,都可能导致下一次捕获卡在等待状态。真正稳定运行的关键不在“怎么拿到帧”,而在“谁在什么时候释放了什么”。

相关专题

更多
视频后缀名都有哪些
视频后缀名都有哪些

视频后缀名都有avi、mpg、mpeg、rm、rmvb、flv、wmv、mov、mkv、ASF、M1V、M2V、MPE、QT、VOB、RA、RMJ、RMS、RAM、等等。更多关于视频后缀名的相关知识,详情请看本专题下面的文章,php中文网欢迎大家前来学习。

3329

2023.10.31

C++ Qt图形开发
C++ Qt图形开发

本专题专注于 C++ Qt框架在图形界面开发中的应用,系统讲解窗口设计、信号与槽机制、界面布局、事件处理、数据库连接与跨平台打包等核心技能,通过多个桌面应用项目实战,帮助学员快速掌握 Qt 框架并独立完成跨平台GUI软件的开发。

67

2025.08.15

C++ 图形界面开发基础(Qt方向)
C++ 图形界面开发基础(Qt方向)

本专题系统讲解 使用 C++ 与 Qt 进行图形界面(GUI)开发的核心技能,内容涵盖 Qt 项目结构、窗口组件、信号与槽机制、事件处理、布局管理、资源管理,以及跨平台编译与打包流程。通过多个小型桌面应用实战案例,帮助学习者掌握从界面设计到功能实现的完整 GUI 开发能力。

40

2025.12.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

988

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

48

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

158

2025.12.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

467

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

83

2025.12.24

excel制作动态图表教程
excel制作动态图表教程

本专题整合了excel制作动态图表相关教程,阅读专题下面的文章了解更多详细教程。

30

2025.12.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.2万人学习

Excel 教程
Excel 教程

共162课时 | 10万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 1.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号