我是 PInvoke 新手,需要一些帮助。
我想使用 PInvoke 访问 IEnumGUID,并且我在pinvoke.net上找到了一个代码块。
[ComImport, Guid("0002E000-0000-0000-C000-000000000046"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown), ComVisible(false)]
public interface IEnumGUID
{
int Next(int celt, [Out, MarshalAs(UnmanagedType.LPArray, SizeParamIndex=0)]Guid[] rgelt);
void Skip(int celt);
void Reset();
[return: MarshalAs(UnmanagedType.Interface)]
IEnumGUID Clone();
}
Run Code Online (Sandbox Code Playgroud)
这里的Clone函数将IEnumGUID作为返回值,而原始C++接口中的函数将其作为输出参数。
HRESULT Clone(
[out] IEnumGUID **ppenum
);
Run Code Online (Sandbox Code Playgroud)
我了解到PInvoke自动将HRESULT转换为COMException,但我不知道PInvoke如何将out参数转换为返回值。
请对此进行一些解释,以便我以后能够正确使用该方法。
pinvoke.net 网站并不完美。它有很多错误,与任何人在代码中使用该声明的可能性成反比。这个几率很低,你发现的声明已经严重损坏了。
这与pinvoke没有任何关系,COM互操作是.NET中的一个显着特征。将参数转换为返回值的能力对于 COM 自动化来说是高度特定的,并且对于用属性修饰的参数来说也是高度特定的[out, retval]。[retval] 向 COM 客户端运行时支持库提供提示,表明参数可以被视为返回值。例如,任何脚本语言都会利用这一点,这很常见。任何 .NET 语言也是如此,除非使用 [PreverseSig] 属性显式抑制该转换。
IEnumGUID::Clone() 的参数没有 [retval] 属性,因此不应像具有该属性那样进行声明。这不是唯一的错误,其他成员需要 [PerserveSig] 属性,因为您需要知道它们的返回值才能正确使用它们。正确的声明是:
[ComImport, InterfaceType(ComInterfaceType.InterfaceIsIUnknown), Guid("0002E000-0000-0000-C000-000000000046")]
public interface IEnumGUID
{
[PreserveSig]
int Next(int celt, [Out, MarshalAs(UnmanagedType.LPArray, SizeParamIndex=0)] Guid[] rgelt, out int pceltFetched);
[PreserveSig]
int Skip(int celt);
[PreserveSig]
int Reset();
void Clone(out IEnumGUID ppenum);
}
Run Code Online (Sandbox Code Playgroud)
COM IEnumXxxx 接口非常常见,仅通过枚举器返回的元素类型来区分。其中一些存在于 .NET Framework 中,例如与此相比。