QDatastream中的qt二进制读取错误

时间:2013-04-09 08:38:39

标签: c++ qt binary precision

我正在读取由传感器生成的二进制文件。我在以不同的精度(32或64)读取浮点数时遇到问题。我可以在MATLAB(64位版本)中读取它们,但Qt(Windows上的32位版本)给出了错误的值。我可以阅读dtmth(请参阅下面的结构)。之后,我为Inf获得了值baseline。实际上,此值为0。如你所见,我改变了MSB(LittleEndian)。如果我保留BigEndian,我会得到0作为基线,但其他值则是错误的。我的桌面是64位。

我检查了字节数,它们是正确的。我认为问题是机器精度。

QDataStream in(&file);

           in.setByteOrder(QDataStream::LittleEndian);
           params p;

           in >> p.filetype;               
           in >> p.projectid;
           in >> p.datamin;               
           in >> p.dtyear;
           in >> p.dtmth;              
           in >> p.baseline;
           in >> p.startfrequ;

其中p是定义为的结构:

    struct params
    {
        quint8 filetype;   
        quint16 projectid;
        double datamin;
        quint16 dtyear;
        quint8 dtmth;    
        float baseline;
        double startfrequ;

    };

我可以在MATLAB中阅读它们。我的matlab是64位版本,我在其中读取数据类型如下:

MATLAB:
        uint8 filetype;   
        uint16 projectid;
        float64 datamin;
        uint16 dtyear;
        uint8 dtmth;    
        float32 baseline;
        float64 startfrequ;

如果我错过了任何细节,请告诉我。

编辑:

阅读文件:

    QString fileName = QFileDialog::getOpenFileName(this, tr("Open File"), QString(),
               tr("Raw Files (*.msr);;All files (*.*)"));

           if (!fileName.isEmpty()) {
               qDebug("Attempting to open file..");
               QFile file(fileName);
               if (!file.open(QIODevice::ReadOnly)) {
                   QMessageBox::critical(this, tr("Error"), tr("Could not open file"));
                   return;
               }

               QDataStream in(&file);

提前多多感谢。

1 个答案:

答案 0 :(得分:3)

您使用的是哪个版本的Qt?如果版本优于Qt 4.6,则默认精度为64位which means Qt will try to read your float 32 as a float 64。您需要使用in.setFloatingPointPrecision ( QDataStream::SinglePrecision);

手动设置精度
       in >> p.filetype;               
       in >> p.projectid;
       in >> p.datamin;               
       in >> p.dtyear;
       in >> p.dtmth;    
       in.setFloatingPointPrecision(QDataStream::SinglePrecision);
       in >> p.baseline;
       in.setFloatingPointPrecision(QDataStream::DoublePrecision);
       in >> p.startfrequ;

从你的评论看来,这似乎是个问题。实际上,如果设置为单精度,并且尝试读取p.dataminp.startfrequ(64位),则数据流将读取它们为32位浮点数。并且不仅p.datamin将是不正确的,而且之后的所有值都是不正确的。

首先,检查我的建议是否在最后一行之后使用

      if(in.status() == QDataStream::ReadCorruptData){
            qDebug() << "still doesnt work";
      }