美文网首页
Qt数据库应用21-数据分组导出

Qt数据库应用21-数据分组导出

作者: feiyangqingyun | 来源:发表于2022-06-09 08:41 被阅读0次

    一、前言

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

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

    二、功能特点

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

    三、体验地址

    1. 体验地址:https://pan.baidu.com/s/1ZxG-oyUKe286LPMPxOrO2A 提取码:o05q 文件名:bin_dataout.zip
    2. 国内站点:https://gitee.com/feiyangqingyun
    3. 国际站点:https://github.com/feiyangqingyun
    4. 个人主页:https://blog.csdn.net/feiyangqingyun
    5. 知乎主页:https://www.zhihu.com/people/feiyangqingyun/

    四、效果图

    4.jpg

    五、相关代码

    #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数据库应用21-数据分组导出

          本文链接:https://www.haomeiwen.com/subject/awopmrtx.html