C#?使用com獲取Windows攝像頭列表
前言
想使用ffmpeg打開攝像頭,需要輸入攝像頭的名稱,而ffmpeg本身的枚舉攝像頭列表功能不是接口
,所以需要用其他方式獲取到設備列表。C++獲取視頻設備列表的方法有不少,但C#獲取視頻設備列表的方法網上提供的解決方案基本都是依賴第三方庫的,為了獲取視頻設備列表而引入一整個視頻庫實在是不太必要。經過思考,Windows的directshow和mediafudation都是基于com的,而且C#對com的支持是很好的,基于上述兩點我們完全可以在C#中直接調用com。
一、定義com接口
我們使用directshow獲取視頻設備列表,由于com的跨語言特性,完全可以直接在C#中調用,而不用通過C++封裝一層dll給C#使用。我們首先定義需要的com對象接口。
static readonly Guid SystemDeviceEnum = new Guid(0x62BE5D10, 0x60EB, 0x11D0, 0xBD, 0x3B, 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); static readonly Guid VideoInputDevice = new Guid(0x860BB310, 0x5D01, 0x11D0, 0xBD, 0x3B, 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); [Flags] enum CDef { None = 0x0, ClassDefault = 0x1, BypassClassManager = 0x2, ClassLegacy = 0x4, MeritAboveDoNotUse = 0x8, DevmonCMGRDevice = 0x10, DevmonDMO = 0x20, DevmonPNPDevice = 0x40, DevmonFilter = 0x80, DevmonSelectiveMask = 0xF0 } [ComImport] [SuppressUnmanagedCodeSecurity] [Guid("3127CA40-446E-11CE-8135-00AA004BB851")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface IErrorLog { [PreserveSig] int AddError([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [In] System.Runtime.InteropServices.ComTypes.EXCEPINFO pExcepInfo); } [ComImport] [Localizable(false)] [SuppressUnmanagedCodeSecurity] [Guid("55272A00-42CB-11CE-8135-00AA004BB851")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface IPropertyBag { [PreserveSig] int Read([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [MarshalAs(UnmanagedType.Struct)] out object pVar, [In] IErrorLog pErrorLog); [PreserveSig] int Write([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [In][MarshalAs(UnmanagedType.Struct)] ref object pVar); } [ComImport] [SuppressUnmanagedCodeSecurity] [Guid("29840822-5B84-11D0-BD3B-00A0C911CE86")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface ICreateDevEnum { [PreserveSig] int CreateClassEnumerator([In][MarshalAs(UnmanagedType.LPStruct)] Guid pType, out IEnumMoniker ppEnumMoniker, [In] CDef dwFlags); }
二、枚舉設備
與directshow流程一樣,調用com枚舉設備即可,本文只展示獲取設備名稱(FriendlyName),獲取其他屬性可以參照c++調用directshow的實現(xiàn)。
/// <summary> /// 枚舉視頻設備 /// </summary> public static IEnumerable<string> Devices { get { IMoniker[] monikers = new IMoniker[5]; var devEnum = Activator.CreateInstance(Type.GetTypeFromCLSID(SystemDeviceEnum)) as ICreateDevEnum; IEnumMoniker moniker; if (devEnum.CreateClassEnumerator(VideoInputDevice, out moniker, 0) == 0) { while (true) { int r = moniker.Next(1, monikers, IntPtr.Zero); if (r != 0 || monikers[0] == null) break; yield return GetName(monikers[0]); foreach (var i in monikers) { if(i!=null) Marshal.ReleaseComObject(i); } } Marshal.ReleaseComObject(moniker); } Marshal.ReleaseComObject(devEnum); } } /// <summary> /// 獲取設備名稱 /// </summary> /// <param name="moniker"></param> /// <returns></returns> static string GetName(IMoniker moniker) { IPropertyBag property; object value; object temp = null; try { Guid guid = typeof(IPropertyBag).GUID; moniker.BindToStorage(null, null, ref guid, out temp); property = temp as IPropertyBag; int hr = property.Read("FriendlyName", out value, null); Marshal.ThrowExceptionForHR(hr); return value as string; } catch (Exception) { return null; } finally { if (temp != null) { Marshal.ReleaseComObject(temp); } } }
三、完整代碼
using System; using System.Collections.Generic; using System.ComponentModel; using System.Runtime.InteropServices; using System.Runtime.InteropServices.ComTypes; using System.Security; namespace AC { public class EnumDevices { /// <summary> /// 枚舉視頻設備 /// </summary> public static IEnumerable<string> Devices { get { IMoniker[] monikers = new IMoniker[5]; var devEnum = Activator.CreateInstance(Type.GetTypeFromCLSID(SystemDeviceEnum)) as ICreateDevEnum; IEnumMoniker moniker; if (devEnum.CreateClassEnumerator(VideoInputDevice, out moniker, 0) == 0) { while (true) { int hr = moniker.Next(1, monikers, IntPtr.Zero); if (hr != 0 || monikers[0] == null) break; yield return GetName(monikers[0]); foreach (var i in monikers) { if(i!=null) Marshal.ReleaseComObject(i); } } Marshal.ReleaseComObject(moniker); } Marshal.ReleaseComObject(devEnum); } } /// <summary> /// 獲取設備名稱 /// </summary> /// <param name="moniker"></param> /// <returns></returns> static string GetName(IMoniker moniker) { IPropertyBag property; object value; object temp = null; try { Guid guid = typeof(IPropertyBag).GUID; moniker.BindToStorage(null, null, ref guid, out temp); property = temp as IPropertyBag; int hr = property.Read("FriendlyName", out value, null); Marshal.ThrowExceptionForHR(hr); return value as string; } catch (Exception) { return null; } finally { if (temp != null) { Marshal.ReleaseComObject(temp); } } } static readonly Guid SystemDeviceEnum = new Guid(0x62BE5D10, 0x60EB, 0x11D0, 0xBD, 0x3B, 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); static readonly Guid VideoInputDevice = new Guid(0x860BB310, 0x5D01, 0x11D0, 0xBD, 0x3B, 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); [Flags] enum CDef { None = 0x0, ClassDefault = 0x1, BypassClassManager = 0x2, ClassLegacy = 0x4, MeritAboveDoNotUse = 0x8, DevmonCMGRDevice = 0x10, DevmonDMO = 0x20, DevmonPNPDevice = 0x40, DevmonFilter = 0x80, DevmonSelectiveMask = 0xF0 } [ComImport] [SuppressUnmanagedCodeSecurity] [Guid("3127CA40-446E-11CE-8135-00AA004BB851")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface IErrorLog { [PreserveSig] int AddError([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [In] System.Runtime.InteropServices.ComTypes.EXCEPINFO pExcepInfo); } [ComImport] [Localizable(false)] [SuppressUnmanagedCodeSecurity] [Guid("55272A00-42CB-11CE-8135-00AA004BB851")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface IPropertyBag { [PreserveSig] int Read([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [MarshalAs(UnmanagedType.Struct)] out object pVar, [In] IErrorLog pErrorLog); [PreserveSig] int Write([In][MarshalAs(UnmanagedType.LPWStr)] string pszPropName, [In][MarshalAs(UnmanagedType.Struct)] ref object pVar); } [ComImport] [SuppressUnmanagedCodeSecurity] [Guid("29840822-5B84-11D0-BD3B-00A0C911CE86")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] interface ICreateDevEnum { [PreserveSig] int CreateClassEnumerator([In][MarshalAs(UnmanagedType.LPStruct)] Guid pType, out IEnumMoniker ppEnumMoniker, [In] CDef dwFlags); } } }
四、使用示例
.net 6.0代碼示例如下
// See https://aka.ms/new-console-template for more information using AC; //枚舉設備 foreach (var i in EnumDevices.Devices) { //打印設備名稱 Console.WriteLine(i); }
效果:
總結
以上就是今天要講的內容,本文介紹了C#直接調用com獲取視頻設備列表的方法,只要知道了com的一些基本原理以及c#和com的關系,很容易就能實現(xiàn)c#直接使用directshow的功能,第三方的庫也是做了類似的工作,定義了完整的directshow的接口,只是筆者使用的環(huán)境中只需要枚舉視頻設備列表,不需要其他功能,引入完整的directshow接口有點大材小用,所以還不如自己定義幾個必要的接口來的實在。
到此這篇關于C# 使用com獲取Windows攝像頭列表的文章就介紹到這了,更多相關C# 獲取Windows攝像頭列表內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
C#中Hashtable和Dictionary的區(qū)別與用法示例
由于 Hashtable 和 Dictionary 同時存在, 在使用場景上必然存在選擇性, 并不任何時刻都能相互替代。所以這篇文章主要給大家介紹了關于C#中Hashtable和Dictionary區(qū)別的相關資料,需要的朋友可以參考下2021-05-05C#實現(xiàn)在兩個數(shù)字之間生成隨機數(shù)的方法
這篇文章主要介紹了C#實現(xiàn)在兩個數(shù)字之間生成隨機數(shù)的方法,在一些特殊場景會用到哦,需要的朋友可以參考下2014-08-08