COM对象C#将MMDeviceEnumerator转换为IMMDeviceEnumerator InvalidCastException

cod*_*296 4 c# com

我没有使用COM Imports的经验,只是在处理对我不起作用的其他人的代码

引发InvalidCastException的代码行:

    IMMDeviceEnumerator deviceEnumerator = (IMMDeviceEnumerator)(new MMDeviceEnumerator());
Run Code Online (Sandbox Code Playgroud)

COM导入:

[Guid("BCDE0395-E52F-467C-8E3D-C4579291692E")]
internal class MMDeviceEnumerator
{
}

[Guid("A95664D2-9614-4F35-A746-DE8DB63617E6"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IMMDeviceEnumerator
{
    [PreserveSig]
    int EnumAudioEndpoints(EDataFlow dataFlow, DEVICE_STATE dwStateMask, out IMMDeviceCollection ppDevices);

    [PreserveSig]
    int GetDefaultAudioEndpoint(EDataFlow dataFlow, ERole role, out IMMDevice ppEndpoint);

    [PreserveSig]
    int GetDevice([MarshalAs(UnmanagedType.LPWStr)] string pwstrId, out IMMDevice ppDevice);

    [PreserveSig]
    int RegisterEndpointNotificationCallback(IMMNotificationClient pClient);

    [PreserveSig]
    int UnregisterEndpointNotificationCallback(IMMNotificationClient pClient);
}
Run Code Online (Sandbox Code Playgroud)

屏幕截图:

    i.stack.imgur.com/SZODi.png
Run Code Online (Sandbox Code Playgroud)

Han*_*ant 5

那不是很接近,您正在创建.NET类。让CLR知道这实际上是一个COM声明,并且已在其他地方实现,需要使用[ComImport]指令。我会给您最少的要求声明:

[ComImport]
[Guid("A95664D2-9614-4F35-A746-DE8DB63617E6")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IMMDeviceEnumerator
{
    // etc..
}

public static class MMDeviceEnumeratorFactory {
    private static readonly Guid MMDeviceEnumerator = new Guid("BCDE0395-E52F-467C-8E3D-C4579291692E");

    public static IMMDeviceEnumerator CreateInstance() {
        var type = Type.GetTypeFromCLSID(MMDeviceEnumerator);
        return (IMMDeviceEnumerator)Activator.CreateInstance(type);
    }
}
Run Code Online (Sandbox Code Playgroud)

并像这样使用它:

IMMDeviceEnumerator deviceEnumerator = MMDeviceEnumeratorFactory.CreateInstance();
Run Code Online (Sandbox Code Playgroud)

强烈避免使用[PreserveSig],当方法失败时,您希望听到一声巨响。请注意,此接口已被NAudio库包装。