Qt数据库应用21-数据分组导出
一、前言
数据分组导出和打印这个需求并不是近期的需求,而是之前做温湿度监控系统的时候提的需求,当然也有几个系统用到了,比如啤酒保鲜监控系统。这个需求的应用场景是,有很多个设备,每个设备都产生了很多的运行日志、报警日志等,这些日志按照时间顺序存储在数据库中,用户需要按照不同设备分组导出,同时对应统计有多少行记录,开始时间和结束时间,以副标题的形式展示在文档中。
数据源有了,关键是如何组织这些数据,传入参数的时候以特定分隔符做标记,取出来生成文档的时候,按照特定分隔符分割字符串,然后循环遍历取出数据,按照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/1ZxG-oyUKe286LPMPxOrO2A 提取码:o05q 文件名:bin_dataout.zip
- 国内站点:https://gitee.com/feiyangqingyun
- 国际站点:https://github.com/feiyangqingyun
- 个人主页:https://blog.csdn.net/feiyangqingyun
- 知乎主页: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-数据分组导出的更多相关文章
- SQL Server 之 在数据库之间进行数据导入导出
1.同一服务器上数据库之间进行数据导入导出 (1).使用 SELECT INTO 导出数据 在SQL Server中使用最广泛的就是通过SELECT INTO语句导出数据,SELECT INTO语句同 ...
- NPOI从数据库中调取数据直接导出到EXCEL中
关于NPOI NPOI是POI项目的.NET版本,是由@Tony Qu(http://tonyqus.cnblogs.com/)等大侠基于POI开发的,可以从http://npoi.codeplex. ...
- 用C# ASP.net将数据库中的数据表导出到Excel中
需要用到组件GridView和一个button即可. 给GridView添加一个数据源, 选择你想要的数据库中的表的字段,添加成功后GridView中就显示数据. 再添加一个button,双击控件添加 ...
- Oracle数据库的创建、数据导入导出
如何结合Sql脚本和PL/SQL Developer工具来实现创建表空间.创建数据库.备份数据库.还原数据库等操作,然后实现Oracle对象创建.导入数据等操作,方便我们快速了解.创建所需要的部署Sq ...
- 使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复
使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复 这种操作百度一搜一大片,今天整理以前做的项目时自己备份了一下数据库,试着将数据进行导出备份和导入恢复了一下:下面是操作过程: 1 ...
- SQLServer2008 和SQLServer2008 R2版本导出 数据库结构和数据sql
①SQLServer2008 版本导出 数据库结构和数据sql ②SQLServer2008R2 版本导出 数据库结构和数据sql 采集 #HUABAN_WIDGETS .HUABAN-red-nor ...
- oracle数据库管理员简介、导入数据与导出数据
数据库管理员: sys和system的权限区别:sys:所有oracle的数据字典的基表和视图都存放在sys用户中,这些基表和视图对于oracle的运行时至关重要的,由数据库 自己维护,任何用户都不能 ...
- JAVA实现数据库数据导入/导出到Excel(POI)
准备工作: 1.导入POI包:POI下载地址http://mirrors.tuna.tsinghua.edu.cn/apache/poi/release/src/(重要) 如下 2.导入Java界面美 ...
- ssm框架之将数据库的数据导入导出为excel文件
在这里首先我要将自己遇到的各种问题,以及需求记录下来,做一个备忘,便于以后查看: 需求:主要实现两个功能,将oracle数据库里的数据导出为excel,同时需要将excel表格的数据导入到数据库 环境 ...
- jsp+servlet+poi导出数据库中的数据
index.jsp: <a href="POIout">简单导出数据</a> <a href="POIoutTemplate"&g ...
随机推荐
- Exchange2016中搜索和删除邮件
Exchange2016中搜索和删除邮件 在以前版本的 Exchange 中,可以运行 Search-Mailbox -DeleteContent 命令搜索并删除电子邮件.你仍可以在 Exchang ...
- 多校A层冲刺NOIP2024模拟赛04
T1.02表示法 竟然有出题人敢出高精度(其实只是一个把string转成01串),开场看出记搜后十分犹豫到底要不要写高精,徘徊很久还是写了个小高精. 码( /* GGrun */ #include&l ...
- 在 Debian 12 上安装 KubeSphere 实战入门
老 Z,运维架构师,云原生爱好者,目前专注于云原生运维,云原生领域技术栈涉及 Kubernetes.KubeSphere.DevOps.OpenStack.Ansible 等. 前言 知识点 定级:入 ...
- Nginx 服务器 SSL 证书安装部署
工具:WinSCP.putty 下载证书 cloud.tencent.com_bundle.crt 证书文件 cloud.tencent.com_bundle.pem 证书文件 cloud.tence ...
- 为Markdown/HTML文档生成一个简易目录
现在阅览文章的网页往往都带有一个目录,方便点击跳转.目录一般都是根据文档中的标题级别直接生成的. 现在我们也来模仿一个简单的,无非就是把<h1><h2>...的序列转化成树嘛 ...
- Vue绘制图片轮播组件【转载】
基本要求:页面加载,自动播放.鼠标悬停,停止播放.鼠标离开,继续播放.点击左右箭头切换上一张,下一张图片.下方小圆点显示当前位第几张图片. 示例代码: 结构html: <template> ...
- 12万字的java面试题及答案整理(2024新版)
前言 本来想着给自己放松一下,刷刷博客,慕然回首,final有哪些用法?static都有哪些用法?java的精度算法?java运算逻辑?异常处理?似乎有点模糊了,那就大概看一下Java基础面试题吧.好记 ...
- C++学习——访问修饰符
一.类是什么 类是C++当中的一个集合,定义了"属性",通过类可以实例化对象,此时对象的属性就囊括在这个类当中.比如: class student { public: string ...
- Django框架表单基础
本节主要介绍一下Django框架表单(Form)的基础知识.Django框架提供了一系列的工具和库来帮助设计人员构建表单,通过表单来接收网站用户的输入,然后处理以及响应这些用户的输入. 6.1.1 H ...
- OSPF协议
OSPF(Open Shortest Path First)开放式最短路径优先,是一种链路状态型路由协议,用于在网络中计算最短路径.OSPF协议是基于Dijkstra算法的,使用链路状态信息来计算最短 ...