最近在项目中使用MediaFoundation连接USB摄像头实现视频播放、录制等功能时,起初使用的是同步读取的方式,即开启一个专用线程,循环读取视频帧并显示。在这种方式下,如果突然把摄像头从计算机中拔出,这个专用线程就会一直停留并卡在读取函数处等待数据。在这种case下,就无法正常结束视频。
转个脑筋想想,MediaFoundation这么官方的platform,不应该有这么明显的bug呀。查了些MediaFoundation的开发文档,符合预期,果然还有异步读取的方式:大多数SDK都会同时提供同步和异步的数据通讯方式,其中同步方式一般较为简单,但是功能受限,而异步方式较为复杂,但是功能往往更强大,更灵活。使用MF的异步读取方式,直接注册一个回调函数,并在回调函数中读取视频帧,同时加入一些超时机制,就可以解决本文所述问题。
关于MediaFoundation的异步读取方法,可以具体参考微软的官方开发文档:
Using the source reader in async mode.
https://msdn.microsoft.com/en-us/library/windows/desktop/gg583871(v=vs.85).aspx
核心读取过程如下:
// Request the first sample. hr = pReader->ReadSample(MF_SOURCE_READER_FIRST_VIDEO_STREAM, 0, NULL, NULL, NULL, NULL); if (FAILED(hr)) { goto done; } while (SUCCEEDED(hr)) { BOOL bEOS;
// 可以将下面的INFINITE改为一个合适的超时时间,解决摄像头被拔出后无响应的问题 hr = pCallback->Wait(INFINITE, &bEOS); if (FAILED(hr) || bEOS) { break; } hr = pReader->ReadSample(MF_SOURCE_READER_FIRST_VIDEO_STREAM, 0, NULL, NULL, NULL, NULL); }