本文接上篇文章 C#获取设备(Audio和Video)名称 简单整理,对第四种方式使用整理.
EnumDevice.dll是网上下载的,也下载了对应的源代码,
对应dll:https://download.csdn.net/download/QQ81867376/12322158
该dll的源码: https://download.csdn.net/download/QQ81867376/12322152
由于项目刚好是x86,所以直接使用上面下载的dll,暂未去编译源代码。
C# 调用EnumDevice.dll的方法时候遇到不少问题,在此记录下。
查看C++函数信息,可以使用工具dllExportViewer
下载地址:http://www.nirsoft.net/utils/dll_export_viewer.html
该dll的原型
__declspec(dllimport)EnumDevice(CAPTURE_DEVICE_TYPE type, char * deviceList[], int nListLen, int & iNumCapDevices);
由于未在前面添加extern "C"
一直找不到该方法,暂时使用了索引来进行。
如果函数过多且经常变化使用索引不恰当 ,但这里只有一个方法,因此无碍,就直接使用索引。
C#方法声明 这里耽误点时间,开始仅仅以为按照类型对上即可,定义了如下方法:
//LPStr、LPWStr、BStr 或 LPTStr [DllImport(EnumDeviceDll, EntryPoint = "#1", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)] public extern static int EnumDevice(CAPTURE_DEVICE_TYPE type, [In, Out, MarshalAs(UnmanagedType.LPArray, ArraySubType = UnmanagedType.LPStr)] string[] deviceList, int nListLen, ref int iNumCapDevices);
public static ListGetDeviceList() { var list = new string[10]; int index = 0; int result = EnumDevice(CAPTURE_DEVICE_TYPE.DSHOW_AUDIO_DEVICE, list, list.Length, ref index); List listAudio = null; if (result == 0) { listAudio = new List (); foreach (var item in list) { if (string.IsNullOrEmpty(item)) { continue; } listAudio.Add(item); } } return listAudio; }
一运行就报错:其他信息: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏 .
无奈 就查看源代码,发现每一项没有指定大小.就添加了代码
for (int i = 0; i < list.Length; i++) { list[i] = new string(new char[256]); }
运行果然没有报错,
可是有乱码和用ffmpeg调用指令 (ffmpeg -list_devices true -f dshow -i dummy) 输出的结果一样乱码.
于是各种转码无效, 各种定义函数 CharSet = CharSet.Ansi,
ArraySubType = UnmanagedType.LPStr //LPStr、LPWStr、BStr 或 LPTStr 。
试了个遍全都无效,无奈只能换思路了。
修改对应类型,C++里面的 char*,正常直接可以用C#里面string或者char *,
之前使用C#调用FFMpeg的API播放rtmp协议的视频和语音的时候 好像使用过,看了下之前代码,重新定义了接口
[DllImport(EnumDeviceDll, EntryPoint = "#1", CallingConvention = CallingConvention.Cdecl)] public extern static int EnumDevice(CAPTURE_DEVICE_TYPE type, [In, Out, MarshalAs(UnmanagedType.LPArray)] IntPtr[] deviceList, int nListLen, ref int iNumCapDevices);
调用赋值
var list = new IntPtr[10]; int index = 0; for (int i = 0; i < list.Length; i++) { list[i] = Marshal.AllocHGlobal(256); }
直接给每一项分配内存,Marshal.AllocHGlobal(256)
结果是 无法或者到底有多个设备列表, 除了正常外 其他全是乱码,
后来就用字符串来代替,
var stringEmpty = new string(new char[256]);
list[i] = Marshal.StringToHGlobalAnsi(stringEmpty);
终于一切都正常了,C#调用一个vc++的函数,就耽误个把小时。