У меня есть беззнаковый char* (байты, отправленные с сервера), который мне нужно преобразовать/преобразовать в массив структур, размер которых точно соответствует размеру беззнакового char*. Без копирования значений, как мне этого добиться? В uchar* может быть любое количество структур.
Как мне этого добиться?
Я пробовал следующее (псевдокод):
// Define the struct:
struct MyStruct:
unsigned char name[6]
long time
double speed
// ...some code...
// Get the uchar*.
unsigned char* chars = getUchars()
// The uchar*'s first byte is a special flag,
// so the structs start at index 1 not 0.
unsigned char* chars2 = &chars[1]
// Convert chars2 to an array of MyStructs.
MyStruct *structs = (MyStruct*) chars2
// Get the first struct and print values.
MyStruct s1 = structs[0]
_print(charArrayToString(s1.name), s1.time, s1.speed)
// When this _print() is called, the struct's
// name value is printed correctly, but its time
// and speed are not printed correctly.
// Get the second struct.
MyStruct s2 = structs[1]
_print(charArrayToString(s2.name), s2.time, s2.speed)
// None of this second struct's name, time and speed
// is printed correctly.
Где я допустил ошибки?
Возможно, вы сможете решить эту проблему, используя специфичную для реализации опцию для создания упакованной структуры.
Создайте функцию dump() в своей структуре, которая выгружает ее по байтам, а также по элементам данных. Что вы, вероятно, обнаружите, так это то, что там есть некоторые отступы, которые вызывают проблемы.
Спасибо за ваши ответы. Что вы подразумеваете под дампом байт за байтом и по члену данных? Вы имеете в виду установить имя, время и скорость каждой структуры по отдельности вместо того, чтобы приводить весь uchar2 непосредственно к MyStruct*?
используйте пакет pragma, он работает в большинстве компиляторов
#include <stdio.h>
struct DefaultPack
{
char str[6];
long time;
double speed;
};
#pragma pack(push,1)
struct Packed
{
char str[6];
long time;
double speed;
};
#pragma pack(pop)
int main (int argc, char *argv[])
{
struct DefaultPack n;
struct Packed p;
printf("Sizes of struct:\n"
"Default packing: %lu\n"
"Packed: %lu\n"
"in packed struct:\n"
"offset of time: %lu\n"
"offset of speed: %lu\n"
"in default packed struct:\n"
"offset of time: %lu\n"
"offset of speed: %lu\n",
sizeof(struct DefaultPack),
sizeof(struct Packed),
(void*)&p.time - (void*)&p.str[0],
(void*)&p.speed - (void*)&p.str[0],
(void*)&n.time - (void*)&n.str[0],
(void*)&n.speed - (void*)&n.str[0]
);
return(0);
}
------ вывод: Размеры структуры: Упаковка по умолчанию: 24 Упаковано: 22 в упакованной структуре: смещение времени: 6 смещение скорости: 14 в упакованной структуре по умолчанию: смещение времени: 8 смещение скорости: 16
long
иdouble
имеют требования к выравниванию.chars2
, вероятно, неправильно выровнен для них.