Qt数据库应用之实现数据分组导出

 更新时间:2022年06月09日 09:00:34   作者:feiyangqingyun  
这篇文章主要为大家详细介绍了如何利用Qt实现数据库数据分组导出,文中的示例代码讲解详细,对我们学习或工作有一定参考价值,需要的可以了解一下

一、前言

数据分组导出和打印这个需求并不是近期的需求,而是之前做温湿度监控系统的时候提的需求,当然也有几个系统用到了,比如啤酒保鲜监控系统。这个需求的应用场景是,有很多个设备,每个设备都产生了很多的运行日志、报警日志等,这些日志按照时间顺序存储在数据库中,用户需要按照不同设备分组导出,同时对应统计有多少行记录,开始时间和结束时间,以副标题的形式展示在文档中。

数据源有了,关键是如何组织这些数据,传入参数的时候以特定分隔符做标记,取出来生成文档的时候,按照特定分隔符分割字符串,然后循环遍历取出数据,按照html格式填充一行行表格内容,最终形成一个完整的html字符串集合,这个字符串集合既可以保存到xls文档,也可以作为打印的内容,导出到pdf就是打印到pdf文件。在分组导出的同时,还可以设置过滤条件,符合特定条件的记录不同颜色显示。

二、功能特点

  • 组件同时集成了导出数据到csv、xls、pdf和打印数据。
  • 所有操作全部提供静态方法无需new,数据和属性等各种参数设置采用结构体数据,极为方便。
  • 同时支持QTableView、QTableWidget、QStandardItemModel、QSqlTableModel等数据源。
  • 提供静态方法直接传入QTableView、QTableWidget控件,自动识别列名、列宽和数据内容。
  • 每组功能都提供单独的完整的示例,注释详细,非常适合各阶段Qter程序员。
  • 原创导出数据机制,不依赖任何office组件或者操作系统等第三方库,支持嵌入式linux。
  • 速度超快,9个字段10万行数据只需要2秒钟完成。
  • 只需要四个步骤即可开始急速导出海量数据比如100W条记录到Excel。
  • 同时提供直接写入数据接口和多线程写入数据接口,不卡主界面。
  • 可设置标题、副标题、表名。
  • 可设置导出数据的字段名、列名、列宽。
  • 可设置末尾列自动拉伸填充,默认拉伸更美观。
  • 可设置是否启用校验过滤数据,启用后符合规则的数据特殊颜色显示。
  • 可指定校验的列、校验规则、校验值、校验值数据类型。
  • 校验规则支持 精确等于==、大于>、大于等于>=、小于<、小于等于<=、不等于!=、包含contains。
  • 校验值数据类型支持 整型int、浮点型float、双精度型double,默认文本字符串类型。
  • 可设置随机背景颜色及需要随机背景色的列集合。
  • 支持分组输出数据,比如按照设备分组输出数据,方便查看。
  • 可设置csv分隔符、行内容分隔符、子内容分隔符。
  • 可设置边框宽度、自动填数据类型,默认自动数据类型开启。
  • 可设置是否开启数据单元格样式,默认不开启,不开启可以节约大概30%的文件体积。
  • 可设置横向排版、纸张边距等,比如导出到pdf以及打印数据。
  • 提供图文混排导出数据到pdf以及打印示例,自动分页,支持多图。
  • 提供一个打印样板中同时包括横向纵向排版示例。
  • 提供静态函数将控件截图导出到pdf文件。
  • 提供静态函数将图片转成pdf文件。
  • 提供静态函数将csv文件转成xls文件,支持列宽表名等参数设置。
  • 针对每列可分别设置字段对齐样式、内容对齐样式,包括左对齐、居中对齐、右对齐。
  • 灵活性超高,可自由更改源码设置对齐方式、文字颜色、背景颜色等。
  • 支持任意excel表格软件,包括但不限于excel2003-2021、wps、openoffice等。
  • 纯Qt编写,支持任意Qt版本+任意编译器+任意系统。

三、体验地址

体验地址:https://pan.baidu.com/s/1eeL5MTz0rifwtVLegRpkoQ  提取码:erxm 文件名:bin_dataout.zip

国内站点:https://gitee.com/feiyangqingyun

国际站点:https://github.com/feiyangqingyun

四、效果图

五、相关代码

#include "frmdataout3.h"
#include "ui_frmdataout3.h"
#include "frmmain.h"
#include "quihelper.h"
#include "dataxls.h"
#include "dataprint.h"

frmDataOut3::frmDataOut3(QWidget *parent) : QWidget(parent), ui(new Ui::frmDataOut3)
{
    ui->setupUi(this);
    this->initForm();
}

frmDataOut3::~frmDataOut3()
{
    delete ui;
}

void frmDataOut3::showEvent(QShowEvent *)
{
    static bool isShow = false;
    if (!isShow) {
        isShow = true;
        QMetaObject::invokeMethod(this, "on_btnLoad_clicked");
    }
}

void frmDataOut3::setInfo(int type, int count)
{
    QString str1 = "生成数据";
    if (type == 1) {
        str1 = "导出数据";
    } else if (type == 2) {
        str1 = "打印数据";
    }

    ui->labInfo1->setText(str1);
    ui->labInfo2->setText(QString("共 %1 条").arg(count));
    QString msec = QString::number((float)time.elapsed() / 1000, 'f', 3);
    ui->labInfo3->setText(QString("用时 %1 秒").arg(msec));
}

void frmDataOut3::getContent(int maxCount, QStringList &content, QStringList &subTitle1, QStringList &subTitle2)
{
    content.clear();
    subTitle1.clear();
    subTitle2.clear();
    maxCount = maxCount >= rowCount ? rowCount : maxCount;

    QString sql = QString("select * from MsgInfo order by DeviceID asc,MsgTime asc limit %1").arg(maxCount);
    QSqlQuery query;
    if (!query.exec(sql)) {
        return;
    }

    int count = 0;
    int logID = 0;
    QString tempDeviceID = "0";
    QString temp = "";
    QString startTime = "";
    QString endTime = "";

    while (query.next()) {
        count++;
        logID++;

        QString deviceID = query.value(0).toString();
        QString deviceName = query.value(1).toString();
        QString deviceTel = query.value(2).toString();
        QString deviceWeight = query.value(3).toString();
        QString deviceTemp = query.value(4).toString();
        QString devicePressure = query.value(5).toString();
        QString msgType = query.value(6).toString();
        QString msgTime = query.value(7).toString();
        QString msgContent = query.value(8).toString();

        //如果是一个全新的设备,则添加子标题,添加上一个设备的数据集合后清空集合数据.
        if (deviceID != tempDeviceID) {
            subTitle1 << QString("设备编号 : %1    设备名称 : %2    设备号码 : %3").arg(deviceID).arg(deviceName).arg(deviceTel);
            //当前设备是一个新的设备,而且当前数据集合不为空,说明上一个设备数据集合已经准备完毕
            //此时添加上一个设备的子标题,因为此时才能计算得到结束时间
            if (!temp.isEmpty()) {
                content << temp.mid(0, temp.length() - 1);
                logID--;

                subTitle2 << QString("开始时间 : %1    结束时间 : %2    记录总数 : %3").arg(startTime).arg(endTime).arg(logID);
                temp = "";
                logID = 1;
            }

            tempDeviceID = deviceID;
            startTime = msgTime;
        }

        endTime =  msgTime;
        temp += QString("%1;%2;%3;%4;%5;%6;%7|").arg(logID).arg(deviceWeight).arg(deviceTemp).arg(devicePressure).arg(msgType).arg(msgTime).arg(msgContent);
    }

    //添加最后一个设备的数据
    if (!temp.isEmpty()) {
        content << temp.mid(0, temp.length() - 1);
        subTitle2 << QString("开始时间 : %1    结束时间 : %2    记录总数 : %3").arg(startTime).arg(endTime).arg(logID);
    }
}

DataContent frmDataOut3::getDataContent(int maxCount)
{
    DataContent dataContent;
    dataContent.title = "所有短信记录";
    dataContent.subTitle = QString("%1 导出短信记录").arg(DATETIME);

    QList<QString> columnNames;
    columnNames << "序号" << "重量" << "温度" << "压力" << "类型" << "接收时间" << "短信内容";
    QList<int> columnWidths;
    columnWidths << 70 << 70 << 70 << 70 << 100 << 180 << 260;
    dataContent.columnNames = columnNames;
    dataContent.columnWidths = columnWidths;

    QStringList content, subTitle1, subTitle2;
    getContent(maxCount, content, subTitle1, subTitle2);
    dataContent.content = content;
    dataContent.subTitle1 = subTitle1;
    dataContent.subTitle2 = subTitle2;

    //默认对齐方式
    dataContent.defaultAlignment = 1;
    //边框粗细
    dataContent.borderWidth = 1;
    //导出到xls有样式比如边框需要开启 cellStyle = true
    //dataContent.cellStyle = true;
    dataContent.randomColor = ui->ckRandomColor->isChecked();

    //下面表示第4列 (对应表中的 类型 字段) 值 包含 重量 关键字 则突出显示
    dataContent.checkColumn = ui->ckCheckColumn->isChecked() ? 4 : -1;
    dataContent.checkType = "contains";
    dataContent.checkValue = "重量";

    dataContent.stretchLast = ui->ckStretchLast->isChecked();
    dataContent.landscape = false;
    return dataContent;
}

void frmDataOut3::initForm()
{
    ui->frame->setFixedWidth(AppConfig::RightWidth);

    columnNames << "设备编号" << "设备名称" << "设备号码" << "重量" << "温度" << "压力" << "类型" << "接收时间" << "短信内容";
    columnWidths << 70 << 150 << 120 << 70 << 70 << 70 << 100 << 180 << 180;

    rowCount = frmMain::getCount();
    columnCount = columnNames.count();

    model = new QSqlTableModel;
    QUIHelper::initTableView(ui->tableView);
}

void frmDataOut3::on_btnLoad_clicked()
{
    time.restart();

    model->setTable("MsgInfo");
    model->setSort(0, Qt::AscendingOrder);
    model->select();
    ui->tableView->setModel(model);

    for (int i = 0; i < columnCount; ++i) {
        model->setHeaderData(i, Qt::Horizontal, columnNames.at(i));
        ui->tableView->setColumnWidth(i, columnWidths.at(i));
    }

    setInfo(0, rowCount);
}

void frmDataOut3::on_btnXls_clicked()
{
    time.restart();

    DataContent dataContent = getDataContent(AppConfig::CountXls);
    dataContent.fileName = QUIHelper::appPath() + "/db/dataout3.xls";
    dataContent.sheetName = "短信信息";
    DataXls::saveXls(dataContent);

    setInfo(1, qMin(rowCount, AppConfig::CountXls));
    QUIHelper::openFile(dataContent.fileName, "导出短信信息");
}

void frmDataOut3::on_btnPdf_clicked()
{
    time.restart();

    DataContent dataContent = getDataContent(AppConfig::CountPdf);
    dataContent.fileName = QUIHelper::appPath() + "/db/dataout3.pdf";
    DataPrint::savePdf(dataContent);

    setInfo(1, qMin(rowCount, AppConfig::CountPdf));
    QUIHelper::openFile(dataContent.fileName, "导出短信信息");
}

void frmDataOut3::on_btnPrint_clicked()
{
    DataContent dataContent = getDataContent(AppConfig::CountPrint);
    DataPrint::print(dataContent);
}

到此这篇关于Qt数据库应用之实现数据分组导出的文章就介绍到这了,更多相关Qt数据分组导出内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++ 函数重载详情介绍

    C++ 函数重载详情介绍

    这篇文章主要介绍了C++ 函数重载详情,函数重载还有一个别名叫函数多态,函数多态是C++在C语言基础上的新特性,它可以让我们使用多个同名函数,下面来看看文章具体内容的介绍
    2021-11-11
  • C++联合体union用法实例详解

    C++联合体union用法实例详解

    这篇文章主要介绍了C++联合体union用法,较为详细的分析了C++中联合体的概念、实用技巧及相关注意事项,需要的朋友可以参考下
    2015-05-05
  • C语言代码链表实现贪吃蛇游戏

    C语言代码链表实现贪吃蛇游戏

    这篇文章主要为大家详细介绍了C语言链表实现贪吃蛇游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-01-01
  • C语言实现密码程序

    C语言实现密码程序

    这篇文章主要为大家详细介绍了C语言实现密码程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-02-02
  • VC++ 2019 "const char*"类型的实参与"LPCTSTR"类型的形参不兼容解决

    VC++ 2019 "const char*"类型的实参与"LPCTSTR"

    这篇文章主要给大家介绍了关于VC++ 2019 "const char*"类型的实参与"LPCTSTR"类型的形参不兼容的解决方法,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2023-03-03
  • C++实现LeetCode(5.最长回文子串)

    C++实现LeetCode(5.最长回文子串)

    这篇文章主要介绍了C++实现LeetCode(5.最长回文子串),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C语言中自动隐式转换与类型强制转换实例分析

    C语言中自动隐式转换与类型强制转换实例分析

    这篇文章主要介绍了C语言中自动隐式转换与类型强制转换实例分析,需要的朋友可以参考下
    2014-07-07
  • C++ 复制控制之复制构造函数的实现

    C++ 复制控制之复制构造函数的实现

    所谓的“复制控制”即通过这三个成员函数控制对象复制的过程,本文主要介绍了C++ 复制控制之复制构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下
    2023-11-11
  • QT5交叉编译入门级教程(arm64、mips64)

    QT5交叉编译入门级教程(arm64、mips64)

    交叉编译就是在当前系统平台上,开发编译运行于其它平台的程序,比如本文硬件环境是x86平台,但是编译出来的程序是在arm64架构、mips64等架构上运行,本文给大家分享QT5交叉编译入门级教程(arm64、mips64),感兴趣的朋友一起看看吧
    2023-11-11
  • C++数据结构链表基本操作示例过程

    C++数据结构链表基本操作示例过程

    这篇文章主要为大家介绍了C++数据结构链表基本操作的示例过程有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2021-11-11

最新评论