一、前言

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

数据源有了,关键是如何组织这些数据,传入参数的时候以特定分隔符做标记,取出来生成文档的时候,按照特定分隔符分割字符串,然后循环遍历取出数据,按照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/

四、效果图

五、相关代码

#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-数据分组导出相关推荐

  1. [转载]在SQL Server数据库之间进行数据导入导出,OPENDATASOURCE

    需要在c盘下先建立一个data.txt文件,然后在文件的第一行写上你要导出的列,不如说要导出id和name这两列,就在第一行写上 id,name 然后保存,使用下列SQL就可以了,你如果要保持原有的I ...

  2. 在SQL Server数据库之间进行数据导入导出

    来源:http://kb.cnblogs.com/page/94464/ 在SQL Server数据库之间进行数据导入导出 (1).使用SELECT INTO导出数据 在SQL Server中使用最广 ...

  3. 数据库查询的数据分组方法

    一.简介. 很多时候,我们进行数据库查询,并不需要清楚的知道每一条数据的全部信息.举个例子,我们想要知道仅仅是,某年每月产生至少两单的客户都有谁,但是,我们使用where语句进行查询的话,查询出来的数 ...

  4. Oracle数据库的创建、数据导入导出

    如何结合Sql脚本和PL/SQL Developer工具来实现创建表空间.创建数据库.备份数据库.还原数据库等操作,然后实现Oracle对象创建.导入数据等操作,方便我们快速了解.创建所需要的部署Sq ...

  5. oracle sql 导入mysql数据库备份_使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复...

    使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复 这种操作百度一搜一大片,今天整理以前做的项目时自己备份了一下数据库,试着将数据进行导出备份和导入恢复了一下:下面是操作过程: 1 ...

  6. 数据库oracle数据导入导出命令,数据库oracle数据导入导出命令

    exp Oracle数据导入导出imp/exp 功能:Oracle数据导入导出imp/exp就相当与oracle数据还原与备份. 大多情况都可以用Oracle数据导入导出完成数据的备份和还原(不会造成 ...

  7. 使用Transact-SQL进行数据导入导出方法详解

    本文为原创,如需转载,请注明作者和出处,谢谢! 本文曾发表于IT168:http://tech.it168.com/db/s/2006-08-16/200608160913336_1.shtml    ...

  8. oracle为什么不用指定数据库,两个数据库怎么保持数据正确显示

    当前位置:我的异常网» Oracle管理 » 两个数据库怎么保持数据正确显示 两个数据库怎么保持数据正确显示 www.myexceptions.net  网友分享于:2015-08-26  浏览:23 ...

  9. easypoi导出数值型_SpringBoot使用EasyPoi进行数据导入导出Excel(一)

    在实际项目开发中,对于Excel的导入导出还是很常见的需求,比如说将数据根据模板批量导入到数据库中,以及将数据库中的数据批量导出陈Excel的形式 现有需求: 下载固定的导入Excel模板 导入Exc ...

最新文章

  1. codeforces 374A Inna and Pink Pony 解题报告
  2. Word中以交叉引用的方式插入参考文献
  3. android仿苹果SwitchButton效果的实现
  4. 542. 01 Matrix
  5. win10编辑js文件报错,错误','
  6. kbengine 的 nginx反向代理https/wss 配置 支持kbe负载均衡
  7. imsi、 ICCID、ki、IMEI
  8. 推荐一个PPT副业,日赚800-1000元
  9. 【使用Idea打包war包】
  10. 数学魔鬼表达式——第二天
  11. Oracle EBS新汇总模板创建系统余额表的记录
  12. idea Translation插件翻译失败。TKK: 更新 TKK 失败,请检查网络连接
  13. 【web前端特效源码】使用HTML5+CSS3制作一个3D视频旋转立方体动画效果~~适合初学者~超简单~ |前端开发|IT软件
  14. 瑜伽修身 偷师动物12招(组图)
  15. python中复制上一行快捷键_Pycharm最常用的快捷键及使用技巧
  16. finereport与finebi差别_一篇文章说清FineReport和FineBI的联系与差别
  17. 吕本富:从平台经济到平台经济学
  18. python_torch_加载数据集_构建模型_构建训练循环_保存和调用训练好的模型
  19. 啊哈c语言有函数么,啊哈c-啊哈c为什么不能运行??如图
  20. json数据和对象的区别

热门文章

  1. Word粘贴快捷键不能用
  2. SharePoint 2013 Search KQL 介绍及实例分析
  3. TrueNAS安装(虚拟机环境)
  4. SAP xsengine 403 - Forbidden(The server refused to fulfill the request.)
  5. Telnet、Mstsc、SSH之间的比较和区别
  6. ios图片放大之后如何不模糊_如何放大图片而不模糊
  7. matlab逐步积分,第17章 隐式逐步积分法.ppt
  8. 计算机应用基础006在线作业2,东师《计算机应用基础》20春在线作业2答案00648
  9. 《开源软件架构》--nginx配置与内部(三)
  10. 圣朱妮佩洛|San Junipero(5)