Маршал С++ struct array в С#
У меня есть следующая структура в С++:
#define MAXCHARS 15
typedef struct
{
char data[MAXCHARS];
int prob[MAXCHARS];
} LPRData;
И функция, в которую я p/вызываю, чтобы получить массив из 3 следующих структур:
void GetData(LPRData *data);
В С++ я бы просто сделал что-то вроде этого:
LPRData *Results;
Results = (LPRData *)malloc(MAXRESULTS*sizeof(LPRData));
GetData( Results );
И все будет хорошо, но в С# я не могу заставить его работать.
Я создал структуру С# следующим образом:
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 (байтовые [] данные);
Но в этом случае я получу строку "данные" из самой первой структуры LPRData, но ничего после нее, включая "prob" массив из одной и той же LPRData.
Любые идеи о том, как правильно справиться с этим?
Ответы
Ответ 1
Я бы постарался добавить некоторые атрибуты к вашей декретации структуры
[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, поэтому я ржавый.
Ответ 2
Один трюк при работе с указателями - это просто использовать 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;
}
Ответ 3
Помощник помощника PInvoke может помочь. http://clrinterop.codeplex.com/releases/view/14120
Ответ 4
Вы отметили параметр GetData с помощью OutAttribute?
Сочетание атрибутов InAttribute и OutAttribute особенно полезен когда применяется к массивам и отформатирован, невоспламеняющиеся. Вызывающие изменения вызываемого лица к этим типам только при применении обоих атрибутов.
Ответ 5
Аналогичная тема обсуждалась на этом вопросе, и один из заключений заключался в том, что параметр CharSet
named должен быть установлен на 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;
}