1.使用DShow获取本机的视音频设备和自带编解码器列表

Posted 快乐阿门

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了1.使用DShow获取本机的视音频设备和自带编解码器列表相关的知识,希望对你有一定的参考价值。

最近做摄像头开发,用opencv做视频的采集工作,但opencv有个问题,采集出来的数据只有视频,不支持音频,而且即使是视频的采集也是封装的DSHOW的采集方案,所以想把之前做的程序换成DShow的。

如果想深入了解DShow,其实DShow安装目录下的开发文档DirectX9_c.chm就足够了,位置在...\\DXSDK\\Doc\\DirectX9下,文档很详细,网上很多博文都是参考或翻译至这个文档。但毕竟看文档比较枯燥,但如果直接说Filter,可能又有些难以理解,不如咱们从一些小例子开始吧。

做摄像头开发第一步其实就是需要获取当前机器连接的图形图像设备(包括本机自带和usb图像设备),用DShow来获取这些设备其实很简单,首先,来获取视频采集设备。主要代码如下:

1.我先定义了一个结构,用来存获取到的设备信息 ASImgDeviceInfoArray m_asVideoDeviceInfo;

这个结构中有一个设备序号,这个序号在DSHOW开发过程中可能没什么用,但使用过opencv的可能知道,打开采集设备是要传序号的,如果一台机器有多个采集设备,哪个序号对应哪个设备,这就很重要了。

struct ImgDeviceInfo

	CString strDevicePidVid;		//设备PIDVID
	CString strDeviceName;			//设备名称
	int nDeviceIndex;				//设备序号

	ImgDeviceInfo()
	
		strDevicePidVid = _T("");
		strDeviceName = _T("");
		nDeviceIndex = -1;
	;
	
	ImgDeviceInfo(const ImgDeviceInfo &other)
	
		*this = other;
	;
	
	ImgDeviceInfo& operator = (const ImgDeviceInfo& other)
	
		strDevicePidVid = other.strDevicePidVid;
		strDeviceName = other.strDeviceName;
		nDeviceIndex = other.nDeviceIndex;
		return *this;
	;
;
typedef CArray <ImgDeviceInfo, ImgDeviceInfo&> ASImgDeviceInfoArray;


2.枚举采集设备

	int count = 0;

	ImgDeviceInfo sDevice;
   // enumerate all video capture devices
	ICreateDevEnum *pCreateDevEnum = NULL;
    HRESULT hr = CoCreateInstance(CLSID_SystemDeviceEnum, NULL, CLSCTX_INPROC_SERVER,
									IID_ICreateDevEnum, (void**)&pCreateDevEnum);

    IEnumMoniker *pEm = NULL;
    hr = pCreateDevEnum->CreateClassEnumerator(CLSID_VideoInputDeviceCategory, &pEm, 0);
    if (hr != NOERROR) 
		return 0;

    ULONG cFetched;
    IMoniker *pM = NULL;
    while(hr = pEm->Next(1, &pM, &cFetched), hr==S_OK)
    
		sDevice.nDeviceIndex = count;

		LPOLESTR strPidvid = NULL;
		hr = pM->GetDisplayName(0, 0, &strPidvid);  //获取ID信息
		if(SUCCEEDED(hr))
		
			//这里获取了一下设备的ID
			USES_CONVERSION; //OLE2T
			CString sPidvid = strPidvid;
			string str = T2A(sPidvid);
			string result;
			static const regex re("(vid_[0-9a-f]4&pid_[0-9a-f]4)",regex::icase);
			smatch match;
			if (regex_search(str, match, re) && match.size() > 1) 
			
				result = match.str(1);
			 
			else 
			
				result = string("");
			 
			CString strPid(result.c_str());
			strPid.MakeUpper(); //全部大写
			sDevice.strDevicePidVid = strPid;

			IPropertyBag *pBag=0;
			hr = pM->BindToStorage(0, 0, IID_IPropertyBag, (void **)&pBag);
			if(SUCCEEDED(hr))
			
				VARIANT var;
				var.vt = VT_BSTR;
				hr = pBag->Read(L"FriendlyName", &var, NULL); //还有其他属性,像描述信息等等...
				if(hr == NOERROR)
				
					//获取设备名称	
					char camera_name[1024]; 
					WideCharToMultiByte(CP_ACP,0,var.bstrVal,-1, camera_name, sizeof(camera_name) ,"",NULL);
					CString str(camera_name);
					sDevice.strDeviceName = str;
					m_asVideoDeviceInfo.Add(sDevice);
					m_cbxCtrl.AddString(str);

					SysFreeString(var.bstrVal);				
				
				pBag->Release();
			
		
		pM->Release();
		
		count++;
    

	if (m_cbxCtrl.GetCount() > 0)
	
		m_cbxCtrl.SetCurSel(0);
	

	pEm->Release();
	pCreateDevEnum->Release();
同理,获取音频采集设备,视音频编解码器,代码都差不多,只要修改下面代码中的第一个参数即可
hr = pCreateDevEnum->CreateClassEnumerator(CLSID_VideoInputDeviceCategory, &pEm, 0);

得到的结果如下图:


详细代码,见这儿代码下载




以上是关于1.使用DShow获取本机的视音频设备和自带编解码器列表的主要内容,如果未能解决你的问题,请参考以下文章

[总结]视音频编解码技术零基础学习方法

[总结]视音频编解码技术零基础学习方法

[总结]视音频编解码技术零基础学习方法

IP传输工作原理

javaCV开发详解之10:基于dshow调用windows摄像头视频和音频,想要获取屏幕画面首选gdigrab

高保真的音频编解码器模块及方案解析