Qt文件读写

做一个简单的文件读写,我们把一个结构体内的数据写入到二进制文件中,并重新读取解析。代码结构如下:

项目名称随便起就好了。main.cpp是主函数;DataHandler实现文件的写与读,还要模拟过程;Definition.h放置自己常用的定义。

DataHandler

头文件的声明如下:

#ifndef DATAHANDLER_H
#define DATAHANDLER_H

#include <QFile>
#include <QDir>
#include <QFile>
#include <QObject>
#include <QDateTime>
#include <QThread>
#include <QDataStream>
#include <QtEndian>
#include <QCoreApplication>
#include "Definition.h"

class DataHandler: public QObject
{
    Q_OBJECT
public:
    explicit DataHandler(QObject *parent = nullptr);
    void writeData(const DataFrame& headNode, const QByteArray& datagram);
    void resetDataFrame();
    void simulateFunc();
    void analyse();

private:
    QFile           _File;
    DataFrame       _DataFrame;
    int             _id;
    QDataStream     _DataStream;
};

#endif // DATAHANDLER_H

其中,成员变量_File用来打开文件,_DataFrame是一个结构体,放置需要写的数据,_DataStream绑定文件后,可以通过DataStream写入数据。writeData()用来往文件中写数据,resetDataFrame()可以重置DataFrame,simulateFunc()进行模拟测试过程;analyse()读取已经保存的.dat文件并解析。具体实现如下:

#include "DataHandler.h"


DataHandler::DataHandler(QObject *parent): _id(0)
{

}

void DataHandler::writeData(const DataFrame& headNode, const QByteArray& datagram)
{
    /*
        QByteArray      _head;
        quint16         _id;
        quint16         _info_length;
        QString         _info;
        quint32         _datagram_length;
    */
    _DataStream.writeRawData(headNode._head.data(), headNode._head.size());
    _DataStream.writeRawData((char*)&headNode._id, sizeof(headNode._id));
    _DataStream.writeRawData((char*)&headNode._info_length, sizeof(headNode._info_length));

    QByteArray info = QByteArray::fromRawData((char*)headNode._info.toUtf8().data(), headNode._info.toUtf8().size());
    _DataStream.writeRawData(info.data(), info.size());

    _DataStream.writeRawData((char*)&headNode._datagram_length, sizeof(headNode._datagram_length));
    _DataStream.writeRawData((char*)&datagram, datagram.size());
}

void DataHandler::resetDataFrame()
{
    /*
        QByteArray      _head;
        quint16         _id;
        quint16         _info_length;
        QString         _info;
        quint32         _datagram_length;
    */

    _DataFrame._head.resize(DATAFRAME_HEAD_SIZE);
    _DataFrame._head[0] = 0x16;
    _DataFrame._head[1] = 0x06;
    _DataFrame._head[2] = 0x00;
    _DataFrame._head[3] = 0x16;

    _DataFrame._id = this->_id++;
    _DataFrame._info = QString::fromLocal8Bit("Test Data...");
    _DataFrame._info_length = _DataFrame._info.size();
    _DataFrame._datagram_length = 0;
}

void DataHandler::simulateFunc()
{
    /* 模拟测试过程 */
    QString filepath = "./SimulateFunc.dat";
    _File.setFileName(filepath);
    bool success = _File.open(QIODevice::WriteOnly);
    if(!success) return;

    _DataStream.setDevice(&_File);
    _DataStream.setByteOrder(QDataStream::LittleEndian);

    int count = 5;
    while(count--) {
        QByteArray datagram;
        datagram.resize(10);

        this->resetDataFrame();
        _DataFrame._datagram_length = datagram.size();
        writeData(_DataFrame, datagram);
        qDebug() << "write datagram...";
        QThread::sleep(1);
    }
    _File.close();
}

void DataHandler::analyse()
{
    QString filepath = "./SimulateFunc.dat";
    QFile file(filepath);
    if(!file.open(QIODevice::ReadOnly)) return ;

    QByteArray datagram = file.readAll();
    file.close();

    if(datagram.isEmpty()) return;

    /*
        QByteArray      _head;
        quint16         _id;
        quint16         _info_length;
        QString         _info;
        quint32         _datagram_length;
    */

    int file_seek = 0;  // 位置指针
    while(file_seek < datagram.size()) {
        int pos = datagram.mid(file_seek, DATAFRAME_HEAD_SIZE).indexOf(_DataFrame._head);
        if(pos == -1) {
            qDebug() << "Can't Find Head!";
            return;
        }
        file_seek += DATAFRAME_HEAD_SIZE;

        quint16 id = qFromLittleEndian<quint16>((char*)datagram.mid(file_seek, sizeof(_DataFrame._id)).data());
        file_seek += sizeof (_DataFrame._id);
        qDebug() << "ID: " << id;

        quint16 info_length = qFromLittleEndian<quint16>((char*)datagram.mid(file_seek, sizeof(_DataFrame._info_length)).data());
        file_seek += sizeof (_DataFrame._info_length);
        qDebug() << "Info Length: " << id;

        QString info = QString::fromLocal8Bit((char*)datagram.mid(file_seek, info_length).data(), info_length);
        file_seek += info_length;
        qDebug() << "Info: " << info;

        quint32 datagram_length = qFromLittleEndian<quint32>((char*)datagram.mid(file_seek, sizeof(_DataFrame._datagram_length)).data());
        file_seek += sizeof (_DataFrame._datagram_length);
        qDebug() << "Datagram Length: " << datagram_length;

        QByteArray datagram_core = datagram.mid(file_seek, datagram_length);
        file_seek += datagram_length;
    }
}

实现过程中有几个点需要注意:1. 因为平台可能有点不一样,我全部设置成小端序的文件读写;2. 尽量全部保持同样的写文件接口,比如使用了writeRawData(),就全部都有这个写; 3. 注意文件指针file_seek位置的移动,不然数据会解析失败。

Definition.h

#ifndef DEFINITION_H
#define DEFINITION_H

#include <QDebug>
#include <QObject>
#include <QtEndian>
#include <QDataStream>

#define DATAFRAME_HEAD_SIZE 4

typedef struct DataFrame
{
    QByteArray      _head;
    quint16         _id;
    quint16         _info_length;
    QString         _info;
    quint32         _datagram_length;
};

#endif // DEFINITION_H

这里包含了一些常见的Qt数据类型。

主函数

主函数定义如下:

#include <QCoreApplication>
#include "DataHandler.h"
#include <QDebug>

int main(int argc, char *argv[])
{
    DataHandler handler;
    handler.simulateFunc();
    handler.analyse();

    qDebug() << "Finished!";
    return 0;
}

模拟的结果如下:

我们假设放入5个节点数据,存入.dat文件后,重新读取并解析,解析完成后退出。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/465325.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

机器学习-绪论

机器学习致力于研究如何通过计算的手段、利用经验来改善系统自身的性能。在计算机系统中&#xff0c;“经验”通常以“数据”的形式存在&#xff0c;因此&#xff0c;机器学习所研究的主要内容&#xff0c;是关于在计算机上从数据中产生“模型”的算法&#xff0c;即“学习算法…

NPM 仓库的超集 JSR 来了!

引言 今天在 Deno 博客中看到了一篇文章&#xff0c;介绍了一个叫 JSR 的包管理注册中心&#xff0c;简单尝试了一下觉得还不错&#xff0c;本文将结合原文章和个人体验对 JSR 进行一个详细的介绍。 在现如今的前端开发中&#xff0c;包管理注册中心 (如 npmjs.com) 扮演着至…

Git——本地使用详解

目录 Git1、开始版本控制1.1、初始化Repository1.2、使目录脱离Git控制 2、把文件交给Git管控2.1、创建文件后交给Git2.2、git add之后再次修改文件2.3、git add "--all"与"."参数区别2.4、把暂存区的内容提交到存储库里存档 3、工作区、暂存区与存储库3.1…

Epuck2机器人固件更新及IP查询

文章目录 前言一、下载固件更新软件包&#xff1a;二、查询机器人在局域网下的IP 前言 前面进行了多机器人编队仿真包括集中式和分布式&#xff0c;最近打算在实物机器人上跑一跑之前的编队算法。但由于Epuck2机器人长时间没使用&#xff0c;故对其进行固件的更新&#xff0c;…

软考高级:软件架构本质和作用概念和例题

作者&#xff1a;明明如月学长&#xff0c; CSDN 博客专家&#xff0c;大厂高级 Java 工程师&#xff0c;《性能优化方法论》作者、《解锁大厂思维&#xff1a;剖析《阿里巴巴Java开发手册》》、《再学经典&#xff1a;《Effective Java》独家解析》专栏作者。 热门文章推荐&am…

【PyTorch】进阶学习:一文详细介绍 torch.save() 的应用场景、实战代码示例

【PyTorch】进阶学习&#xff1a;一文详细介绍 torch.save() 的应用场景、实战代码示例 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程…

Vue组件封装方案对比——v-if方式与内置component方式

近期在准备搭建一个通用组件库&#xff0c;而公司现有的各个系统也已有自己的组件库只是没抽离出来&#xff0c;但是目前有两套不同的组件封装方案&#xff0c;所以对于方案的选择比较困惑&#xff0c;于是对两种方式进行了对比&#xff0c;结合网上找到的一些开源组件库进行分…

wireshark解析https数据包

Debian11环境&#xff1a; 在linux环境下抓取访问某个https的网址时抓取的数据包都是加密的&#xff0c;导致无法跟踪到数据包流&#xff0c;现在尝试将抓取的https包进行解密。 1、解密https数据包需要设置SSLKEYLOGFILE变量&#xff0c;推荐写入配置文件中。 echo "exp…

Mysql的行级锁

MySQL 中锁定粒度最小的一种锁&#xff0c;是 针对索引字段加的锁 &#xff0c;只针对当前操作的行记录进行加锁。 行级锁能大大减少数据库操作的冲突。其加锁粒度最小&#xff0c;并发度高&#xff0c;但加锁的开销也最大&#xff0c;加锁慢&#xff0c;会出现死锁。行级锁和存…

Ps:文字工具

工具箱里的文字工具组中包含了四种工具&#xff1a; 横排文字工具 Horizontal Type Tool 直排文字工具 Vertical Type Tool 横排文字蒙版工具 Horizontal Type Mask Tool 直排文字蒙版工具 Vertical Type Mask Tool 快捷键&#xff1a;T 横排文字蒙版工具和直排文字蒙版工具…

C++第六弹---类与对象(三)

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】 目录 1、类的6个默认成员函数 2、构造函数 2.1、概念 2.2、特性 3、析构函数 3.1、概念 3.2、特性 3.3、调用顺序 总结 1、类的6个默认成员函数…

力扣hot100:33. 搜索旋转排序数组(二分的理解)

33.搜索旋转排序数组 ​ 这是一个非常有趣的问题&#xff0c;如果不要求使用O(logn)应该没人会想到吧。。 方法一&#xff1a; 极致的分类讨论。旋转排序数组&#xff0c;无非就是右边的增区间的数小于左边的增区间的数&#xff0c;然后依次排序。因此我们只需要分三类讨论即可…

【测试开发学习历程】MySQL数据类型 + MySQL表创建与操作

前言&#xff1a; 半夜梦到自己没有写今天的博客&#xff0c;结果惊醒起来看一看。 得&#xff0c;真的没写。QWQ 可谓垂死病中惊坐起了。 看看发博的时间6&#xff1a;16&#xff0c;而不是什么整点的&#xff0c;就知道我4点就起来了&#xff0c;不是定时发布&#xff01…

知识积累(五):Transformer 家族的学习笔记

文章目录 1. RNN1.1 缺点 2. Transformer2.1 组成2.2 Encoder2.2.1 Input Embedding&#xff08;嵌入层&#xff09;2.2.2 位置编码2.2.3 多头注意力2.2.4 Add & Norm 2.3 Decoder2.3.1 概览2.3.2 Masked multi-head attention 2.4 Transformer 模型的训练和推理2.4.1 训练…

C语言学习过程总结(16)——指针(4)

一、数组名的理解 我们直接使用%p打印出地址来看看&arr【0】 和 arr的不同&#xff1a; int main() {int arr[10] { 1,2,3,4,5,6,7,8,9,10 };printf("&arr[0] %p\n", &arr[0]);printf("arr %p\n", arr);} 、 很容易看出来两者的输出…

ES模块化

Node.js默认并不支持ES模块化&#xff0c;如果需要使用可以采用两种方式。方式一&#xff0c;直接将所有的js文件修改为mjs扩展名。方式二&#xff0c;修改package.json中type属性为module。 导出 默认导出 // 向外部导出内容 export let a 10 export const b "孙悟空…

数据分析 | NumPy

NumPy&#xff0c;全称是 Numerical Python&#xff0c;它是目前 Python 数值计算中最重要的基础模块。NumPy 是针对多维数组的一个科学计算模块&#xff0c;这个模块封装了很多数组类型的常用操作。 使用numpy来创建数组 import numpy as npdata np.array([1, 2, 3]) print…

Unity中UGUI中的PSD导入工具的原理和作用

先说一下PSD导入工具的作用&#xff0c;比如在和美术同事合作开发一个背包UI业务系统时&#xff0c;美术做好效果图后&#xff0c;程序在UGUI中制作好界面&#xff0c;美术说这个图差了2像素&#xff0c;那个图位置不对差了1像素&#xff0c;另外一个图大小不对等等一系列零碎的…

文件包含漏洞(input、filter、zip)

一、PHP://INPUT php://input可以访问请求的原始数据的只读流&#xff0c;将post请求的数据当作php代码执行。当传入的参数作为文件名打开时&#xff0c;可以将参数设为php://input,同时post想设置的文件内容&#xff0c;php执行时会将post内容当作文件内容。从而导致任意代码…

ngnix安装配置

通过yum -y install nginx的方式&#xff0c;有时候会出现No package nginx available的报错。迟迟无法解决。此时要通过下载安装包的方式安装。 1、下载安装包&#xff1a;官方网址 2、解压缩&#xff1a; tar -xzvf nginx-1.23.4.tar.gz cd nginx-1.23.4.tar.gz 3、源码包…