crossplatform.ru

Здравствуйте, гость ( Вход | Регистрация )

 
Ответить в данную темуНачать новую тему
> Проблема с чтением в структуру
x-8973
  опции профиля:
сообщение 28.3.2014, 9:52
Сообщение #1


Студент
*

Группа: Участник
Сообщений: 25
Регистрация: 22.2.2013
Пользователь №: 3720

Спасибо сказали: 0 раз(а)




Репутация:   0  


Доброго здравия всем.
Задача: прочитать из бинарного файла последовательность байт и распихать ее в поля структуры.
Проблема: определил структуру как тип, описал переменную этого типа. Чтение из файла проходит нормально, в переменной типа QByteArray лежат считанные байты. Пытаюсь "наложить" адрес переменной-структуры на адрес переменной-массива:
Widget.h
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QByteArray>
#include <QTextStream>
#include <QFile>
#include <QFileDialog>
#include <QMessageBox>

#pragma pack(push, 1)

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
    typedef struct
    {
        quint8 sync1;
        quint8 sync2;
        quint8 length;
        quint8 packID;
        quint16 status;
        quint16 counter;
        float dVx;
        float dVy;
        float dVz;
        float dWx;
        float dWy;
        float dWz;
        float TAx;
        float TAy;
        float TAz;
        float TGx;
        float TGy;
        float TGz;
        quint16 CRC;
    } packStruct;

    quint16 getCRC(QByteArray package);

private slots:
    void on_pushButton_clicked();

private:
    Ui::Widget *ui;
};

#pragma pack(pop)

#endif // WIDGET_H

Сам код:
QByteArray packet = binaryFile.read(sizeof(packStruct));
packStruct *dataPackage = (packStruct*)packet.data();

В результате на входе (в переменной-массиве) считанные данные появляются, а в переменной-структуре - каша. Что не так? Может, я что-то упустил?
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
ilyabvt
  опции профиля:
сообщение 28.3.2014, 16:56
Сообщение #2


Активный участник
***

Группа: Участник
Сообщений: 297
Регистрация: 23.6.2011
Пользователь №: 2765

Спасибо сказали: 45 раз(а)




Репутация:   3  


Размер структуры не обязательно равен сумме размеров всех полей. Компилятор может выравнивать поля.
Можно отключить выравнивание в компиляторе или использовать нестандартные ключевые слова типа "packed".
Можно писать в файл именно структуру, но есть нюанс: если скомпилировать 2 экземпляра приложения (сплошь и рядом в опенсорсе), с выравниванием и без него, то файлы будут бинарно несовместимы.
Можно читать отдельно по полям.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
x-8973
  опции профиля:
сообщение 28.3.2014, 19:30
Сообщение #3


Студент
*

Группа: Участник
Сообщений: 25
Регистрация: 22.2.2013
Пользователь №: 3720

Спасибо сказали: 0 раз(а)




Репутация:   0  


Проблему удалось решить следующим способом:
Описание структуры вынес из класса;
Считал данные напрямую в переменную-структуру с помощью перегруженного метода quint64 QFile::read(const char* s, uint64 len);
Код сейчас не смогу привести, ибо проект остался на рабочей машине.
ilyabvt, за выравнивание отвечают директивы #pragma pack(push, 1) и #pragma pack(pop) в заголовочнике. Файл пишется абсолютно другой программой, написанной в Qt, интегрированном в ВижлСтудию. На удивление, там чтение написанным мной изначально способом вполне нормально работает. Сегодня полдня всем отделом голову ломали, что не так.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
Iron Bug
  опции профиля:
сообщение 28.3.2014, 20:34
Сообщение #4


Профессионал
*****

Группа: Модератор
Сообщений: 1611
Регистрация: 6.2.2009
Из: Yekaterinburg
Пользователь №: 533

Спасибо сказали: 219 раз(а)




Репутация:   12  


а как и где используется структура? вообще, с потоковыми данными и указателями на динамические структуры надо быть осторожным. они существуют только на момент вызова. при любом изменении объекта они становятся невалидны.
ну и у разных компиляторов могут быть разные схемы по представлению разных типов данных.

Сообщение отредактировал Iron Bug - 28.3.2014, 20:35
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
x-8973
  опции профиля:
сообщение 29.3.2014, 7:30
Сообщение #5


Студент
*

Группа: Участник
Сообщений: 25
Регистрация: 22.2.2013
Пользователь №: 3720

Спасибо сказали: 0 раз(а)




Репутация:   0  


Цитата(Iron Bug @ 28.3.2014, 23:34) *
а как и где используется структура?

Потом подсчитывается CRC пакета данных, и если сумма совпала с записанной в пакете, то некоторые поля сохраняются в текстовик.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
Iron Bug
  опции профиля:
сообщение 29.3.2014, 11:56
Сообщение #6


Профессионал
*****

Группа: Модератор
Сообщений: 1611
Регистрация: 6.2.2009
Из: Yekaterinburg
Пользователь №: 533

Спасибо сказали: 219 раз(а)




Репутация:   12  


Цитата(x-8973 @ 29.3.2014, 10:30) *
Потом подсчитывается CRC пакета данных, и если сумма совпала с записанной в пакете, то некоторые поля сохраняются в текстовик.

да пофиг на семантику. имеется в виду как идёт работа с переменной. сразу после присвоения или как-то ещё.
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение
Алексей1153
  опции профиля:
сообщение 4.4.2014, 19:03
Сообщение #7


фрилансер
******

Группа: Участник
Сообщений: 2939
Регистрация: 19.6.2010
Из: Обливион
Пользователь №: 1822

Спасибо сказали: 215 раз(а)




Репутация:   34  


x-8973, на всякий случай замечу: не нужно выравнивать всё подряд, достаточно только нужную "жёсткую" часть, а именно твою железную структуру


namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT



#pragma pack(push,1)
struct packStruct
    {
        quint8 sync1;
        quint8 sync2;
        quint8 length;
        quint8 packID;
        quint16 status;
        quint16 counter;
        float dVx;
        float dVy;
        float dVz;
        float dWx;
        float dWy;
        float dWz;
        float TAx;
        float TAy;
        float TAz;
        float TGx;
        float TGy;
        float TGz;
        quint16 CRC;
    };
#pragma pack(pop)


...
...



x-8973, как вариант быстрой установки диагноза: сохранить дампы отправленных и принятых массивов в обеих программах (я так понял, их у вас два варианта). Естественно, они должны совпадать :) Если это не так - проанализировать, где покривилось. Результаты в студию
Перейти в начало страницы
 
Быстрая цитата+Цитировать сообщение

Быстрый ответОтветить в данную темуНачать новую тему
Теги
Нет тегов для показа


1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0




RSS Текстовая версия Сейчас: 19.4.2024, 10:45