Qt处理JSON数据(含数组)并保存至excel表格

今天主要是说一下,今天在项目中解决掉得一个项目需求,需求大概是这样的:qt后台需要处理一份JSON文件,当然了这个是一份标准的JSON文件,然后处理转换存入excel表格。其实这个需求可以分解为三步:

  1. 解析JSON文件中的数据
  2. 将第一步中解析出来的数据保存在数据结构中。
  3. 将第二部中的保存的数据写入excel表格。
    为了能够让大家更方便的学习,我写了一个小的项目,来让大家看的更清楚。具体的项目是做的过程是将下面这一份示例文件,存入excel表格。
{
    "NBA": [{
        "teamName": "金州勇士队",
        "coach": "史蒂夫-科尔",
        "boss": "乔-拉克布"
    }, {
        "teamName": "洛杉矶快船",
        "coach": "道格-里弗斯",
        "boss": "史蒂夫-鲍尔默"
    }, {
        "teamName": "洛杉矶湖人",
        "coach": "卢克-沃顿",
        "boss": "珍妮-巴斯"
    }, {
        "teamName": "萨克拉门托国王",
        "coach": "戴夫-乔尔格",
        "boss": "维维克-拉纳戴夫"
    }, {
        "teamName": "波特兰开拓者",
        "coach": "特里-斯托茨",
        "boss": "保罗-艾伦"
    }, {
        "teamName": "俄克拉荷马雷霆",
        "coach": "比利-多诺万",
        "boss": "克莱顿-本内特"
    }]
}

最后保存在excel表格中的样式是这样的:


excel样式

下面开始说说实现这个小项目的代码。

1.解析JSON文件中的数据。

这一步中,虽然说qt都有现成的API供大家调用,不过我在这一块还是花了不少时间,当然这只是对比第二三步来说,或许是因为我对JSON格式的数据不是很清楚吧,这块的API还是挺绕的。

bool MainWindow::analysisJson(QString FileName)
{
    //采用普通方式打开文件,并且存入allDada中,注意这是一种QByteArray格式
    QFile loadFile(FileName);
    if(!loadFile.open(QIODevice::ReadOnly))
    {
        qDebug() << "could't open projects json";
        return false;
    }
    QByteArray allData = loadFile.readAll();
    loadFile.close();
    //开始进行一系列JSON相关的处理
    QJsonParseError json_error;
    QJsonDocument jsonDoc(QJsonDocument::fromJson(allData, &json_error));
    if(json_error.error != QJsonParseError::NoError)
    {
        qDebug() << "json error!";
        return false;
    }
    QJsonObject rootObj = jsonDoc.object();
    //最后数据存入了QJsonObject格式的rootObj中
    if (rootObj.contains("NBA"))
    {
        QJsonValue value = rootObj.value("NBA");
        if (value.isArray())
        {
            QJsonArray array = value.toArray();
            int nSize = array.size();
            for (int i = 0; i < nSize; ++i)
            {
                NBATeamData temp;
                temp.teamName = array.at(i).toObject().value("teamName").toString();
                temp.coachName = array.at(i).toObject().value("coach").toString();
                temp.BossName = array.at(i).toObject().value("boss").toString();
                teamData.append(temp);
            }
        }
    }
}
  1. 这部分的代码实现从开始到处理rootObj之前,这其实都是套路,大家只需要照搬过来就可以了。这部分也没有啥需要说的。
  2. 需要注意的就是我处理rootObj的思路。首先分析这个JSON文件,其实很简单,(大家都知道JSON文件说白了就是键值对),这个文件就是一个键值对嵌套了另外一个键值对,然后里边的键值对还用数组表示。这是一个比较难的一个点。
    2.1 首先判断并且根据最外层的键值对的key值,来获取QJsonValue对象。这个对象是存储着值的。
    2.2 然后根据判断2.1中获取到的QJsonValue对象是不是数组结构,是的话,转成QJsonArray.
    2.3 这一步的思路是最重要的,2.2的做法就是为了遍历,然后因为数组中的每一个值,又是三个键值对。想要实现这三个键值对的获取,就必须将之在转化为Object对象,然后再按照处理Object的方法去处理。

2. 解析出来的数据保存在数据结构中

下面是我定义的用来储存数据结构的一个类。

class NBATeamData
{
public:
    QString teamName;
    QString coachName;
    QString BossName;
};

其实我是用QList<NBATeamData> teamData;来保存JSON的解析结果,也就是定义NBATeamData这个类来存单个数据,然后用QList将其封装起来,以方便操作数据。

3. 写入excel表格

先直接上代码

bool MainWindow::addToExcel(QList<NBATeamData> data)
{
    QString filepath = "D:\\NBA.xls";
    if(!filepath.isEmpty()){
        QAxObject *excel = new QAxObject(this);
        excel->setControl("Excel.Application");//连接Excel控件
        excel->dynamicCall("SetVisible (bool Visible)","false");//不显示窗体
        excel->setProperty("DisplayAlerts", false);//不显示任何警告信息。如果为true那么在关闭是会出现类似“文件已修改,是否保存”的提示
        QAxObject *workbooks = excel->querySubObject("WorkBooks");//获取工作簿集合

        workbooks->dynamicCall("Add");//新建一个工作簿
        QAxObject *workbook = excel->querySubObject("ActiveWorkBook");//获取当前工作簿
        QAxObject *worksheets = workbook->querySubObject("Sheets");//获取工作表集合
        QAxObject *worksheet = worksheets->querySubObject("Item(int)",1);//获取工作表集合的工作表1,即sheet1
        QAxObject *cellA,*cellB,*cellC;

        //设置标题
        int cellrow=1;
        QString A="A"+QString::number(cellrow);//设置要操作的单元格,如A1
        QString B="B"+QString::number(cellrow);
        QString C="C"+QString::number(cellrow);

        //获取单元格
        cellA = worksheet->querySubObject("Range(QVariant, QVariant)",A);
        cellB = worksheet->querySubObject("Range(QVariant, QVariant)",B);
        cellC=worksheet->querySubObject("Range(QVariant, QVariant)",C);

        //设置单元格的值
        cellA->dynamicCall("SetValue(const QVariant&)",QVariant("队名"));
        cellB->dynamicCall("SetValue(const QVariant&)",QVariant("教练"));
        cellC->dynamicCall("SetValue(const QVariant&)",QVariant("老板"));
        cellrow++;

        int rows=data.size();
        for(int i=0;i<rows;i++){
            QString A="A"+QString::number(cellrow);//设置要操作的单元格,如A1
            QString B="B"+QString::number(cellrow);
            QString C="C"+QString::number(cellrow);

            cellA = worksheet->querySubObject("Range(QVariant, QVariant)",A);//获取单元格
            cellB = worksheet->querySubObject("Range(QVariant, QVariant)",B);
            cellC = worksheet->querySubObject("Range(QVariant, QVariant)",C);

            cellA->dynamicCall("SetValue(const QVariant&)",QVariant(data.at(i).teamName));//设置单元格的值
            cellB->dynamicCall("SetValue(const QVariant&)",QVariant(data.at(i).coachName));
            cellC->dynamicCall("SetValue(const QVariant&)",QVariant(data.at(i).BossName));

            cellrow++;
        }

            workbook->dynamicCall("SaveAs(const QString&)",QDir::toNativeSeparators(filepath));//保存至filepath,注意一定要用QDir::toNativeSeparators将路径中的"/"转换为"\",不然一定保存不了。
            workbook->dynamicCall("Close()");//关闭工作簿
            excel->dynamicCall("Quit()");//关闭excel
            delete excel;
            excel=NULL;
        }
        return true;
}
  1. 首先说明一下在QT环境下运行这段代码,要加入如下:其中第一行为头文件,直接加入该文件即可。第二行需要加入.pro文件,加上axcontainer
#include<QAxObject>

QT       += core gui axcontainer

2.之后的话就是一些流水线的代码,相信你对它是不在话下的

代码下载地址:https://download.csdn.net/download/qq130106486/10707414
点击下载代码

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • 关于Mongodb的全面总结 MongoDB的内部构造《MongoDB The Definitive Guide》...
    中v中阅读 31,920评论 2 89
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,028评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,711评论 2 59
  • 累加的可以说"滚雪球" 那么遗忘的呢? 可不可以说"滚时间"? 是增也是减呀! 当有一天雪会溶化, 时间快定格, ...
    透明的蝶阅读 229评论 2 1
  • 请别爱上天蝎座女人 ,天蝎座没有安全感,也不会给别人安全感 请别爱上天蝎座女人 ,天蝎座喜欢折磨人,一会儿对你好,...
    兰公主阅读 1,874评论 2 15