【问题标题】:c# P/Invoke : Pass Enum value with IntPtr to Function; AccessViolationExceptionc# P/Invoke : 将带有 IntPtr 的 Enum 值传递给 Function; AccessViolationException
【发布时间】:2017-12-09 00:14:34
【问题描述】:

我正在实现 C# 包装器以链接 C 库,并且无法将 Enum 值作为参数传递给将指针作为输入的非托管函数调用。那就是抛出这个异常。代码如下所示:

库函数:

int function(infoField Field, void const *value); 

C# 中的元帅:

[DllImport("CLibrary.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int function(infoField Field, IntPtr value);

infoField 结构:

public enum infoField {section, options, orientation};

一个枚举类型“选项”有 4 个值可供选择:

public enum Options { var1, var2, var3, var4};

我想将 4 个选项之一作为 EnumField = Options 的值传递,如下所示:

IntPtr value = (Intptr)Options.var1;
function (infoField.options,  value);

我认为我缺少指向 Enum 类型的指针的正确强制转换约定。将枚举值与 IntPtr 一起传递给函数的正确格式应该是什么? 到目前为止,我尝试了几个选项,但在函数调用时得到 AccessViolationException,这意味着内存分配失败或与库所期望的类型不匹配。

PS:其他 EnumFields 的函数封送处理是成功的。这也是因为值传递对他们来说是字符串值。但特别是对于“选项”,它必须是 4 个枚举选项之一。

【问题讨论】:

  • EnumField 是如何定义的?
  • @TommasoBelluzzo:我已经更新了 *infoField!

标签: c# c++ enums pinvoke marshalling


【解决方案1】:

通常,C++ 枚举数定义如下:

enum class ENUM_FIELD
{ 
    VAR1, 
    VAR2,
    VAR3,
    VAR4
};

默认情况下,枚举类类型的大小为int,这意味着您可以将它们转换为C#,如下所示:

public enum EnumField
{ 
    VAR1, 
    VAR2,
    VAR3,
    VAR4
}

在这种情况下,两个枚举都将在互操作环境中毫无问题地映射。但是如果C++ 枚举以不同的方式定义,或者如果非托管方法需要不同的类型,您可能需要更改C# 声明中的基础类型才能正确映射值。例如:

public enum EnumField : short
{ 
    VAR1, 
    VAR2,
    VAR3,
    VAR4
}

无论如何,我认为问题不是由枚举引起的,而是由您作为const void* 参数传递的内容引起的。将枚举值转换为 IntPtr 对我来说不合适:

UInt32 field = (UInt32)EnumField.VAR1;
GCHandle handle = GCHandle.Alloc(field, GCHandleType.Pinned);
IntPtr ptr = handle.AddrOfPinnedObject();

// pass the ptr variable to the unmanaged function
// and don't forget to handle.Free() when you are done

【讨论】:

  • 谢谢你!我尝试过 int 类型但给出了同样的错误!没抓到这个。
猜你喜欢
  • 2012-09-12
  • 2011-04-17
  • 2013-10-19
  • 2020-02-27
  • 1970-01-01
  • 2013-03-04
  • 1970-01-01
  • 2010-11-06
相关资源
最近更新 更多