将Marshal C ++ struct数组转换为C#
-
06-07-2019 - |
题
我在C ++中有以下结构:
#define MAXCHARS 15
typedef struct
{
char data[MAXCHARS];
int prob[MAXCHARS];
} LPRData;
我正在调用一个函数来获取这些结构中的3个数组:
void GetData(LPRData *data);
在C ++中我会做这样的事情:
LPRData *Results;
Results = (LPRData *)malloc(MAXRESULTS*sizeof(LPRData));
GetData( Results );
它会工作得很好,但在C#中我似乎无法让它工作。 我已经创建了一个像这样的C#结构:
public struct LPRData
{
/// char[15]
[MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 15)]
public string data;
/// int[15]
[MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 15)]
public int[] prob;
}
如果我初始化其中3个(及其所有子数组)的数组并将其传递给它:
GetData(LPRData[] data);
它成功返回,但LPRData数组中的数据没有改变。
我甚至尝试创建一个大小为3 LPRData的原始字节数组,并将其传递给这样的函数原型:
GetData(byte [] data);
但在这种情况下,我会得到“数据”。来自第一个LPRData结构的字符串,但在它之后没有任何内容,包括“prob”结构。来自相同LPRData的数组。
有关如何正确处理此事的任何想法?
解决方案
我会尝试为你的struct decloration添加一些属性
[StructLayout(LayoutKind.Sequential, Size=TotalBytesInStruct),Serializable]
public struct LPRData
{
/// char[15]
[MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 15)]
public string data;
/// int[15]
[MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 15)]
public int[] prob;
}
*注意TotalBytesInStruct不代表变量
JaredPar也是正确的,使用IntPtr类可能会有所帮助,但是因为我使用了PInvoke所以我已经很久了。所以我生锈了。
其他提示
处理指针时的一个技巧就是使用IntPtr。然后,您可以在指针上使用Marshal.PtrToStructure,并根据结构的大小递增以获得结果。
static extern void GetData([Out] out IntPtr ptr);
LPRData[] GetData()
{
IntPtr value;
LPRData[] array = new LPRData[3];
GetData(out value);
for (int i = 0; i < array.Length; i++)
{
array[i] = Marshal.PtrToStructure(value, typeof(LPRData));
value += Marshal.SizeOf(typeof(LPRData));
}
return array;
}
PInvoke Interop助手可能会有所帮助。 http://clrinterop.codeplex.com/releases/view/14120
您是否使用 OutAttribute 一>
结合InAttribute和 OutAttribute特别有用 当应用于数组并格式化时, 不易受伤的类型。来电者看到了 将被调用者更改为这些类型 仅当您应用这两个属性时。
在这个问题上讨论了一个类似的主题,其中一个结论是 CharSet
命名参数必须设置为 CharSet.Ansi
。否则,我们将生成 wchar_t
数组而不是 char
数组。因此,正确的代码如下:
[Serializable]
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct LPRData
{
[MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 15)]
public string data;
[MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 15)]
public int[] prob;
}