PInvoke可以将[out]参数转换为返回值吗?

Kag*_*ght 3 c# pinvoke

我是 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参数转换为返回值

请对此进行一些解释,以便我以后能够正确使用该方法。

Han*_*ant 5

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 中,例如与此相比。