热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

通俗易懂玩QT:Qt使用QAxObject类读写Excel文档(内附源码)

Qt使用QAxObject类读写Excel文档因为工作需要,最近研究了一下使用Qt的QAxObject类实现对Excel文档的简单读写,查阅了很多资料
Qt 使用 QAxObject 类读写 Excel 文档

因为工作需要,最近研究了一下使用 Qt 的 QAxObject 类实现对 Excel 文档的简单读写,查阅了很多资料后,自己总结了一个 ExcelHandle 类出来,下面是实验效果图以及部分核心源码,详细源码请见文章底部,代码注释还算比较详细,细致看看基本都还是很好懂的,如果不知道函数是来干嘛的,可以直接 copy 函数名称到有道翻译,按大写字母隔开即可。
在这里插入图片描述

实验环境

在这里插入图片描述

实验核心源码


excelHandle.h 文件

#ifndef EXCELHANDLE_H
#define EXCELHANDLE_H#include
#include
#include
#include
#include
#include
#include class ExcelHandle : public QObject {Q_OBJECT
public:explicit ExcelHandle(QObject *parent &#61; nullptr);~ExcelHandle();void init(bool showWindow, bool showAlert, QString title);void save(QString url);void exit();void openBook(QString url);void openSheet(int n);void setSheetName(QString name);void addSheet(QString name);void changeSheet(QString name);void changeSheet(int index);void deleteSheet(QString name);void deleteSheet(int index);void writeData(int col, int row, QVariant data);void writeDataSet(QString start_Addr, QList<QList<QVariant>> data_Set);QList<QList<QVariant>> readData();private:QAxObject *excel,*workbooks,*workbook,*sheets,*sheet,*usedRange;QString getEndAddr(QString start_Addr, int col, int row);
};#endif // EXCELHANDLE_H

excelHandle.cpp 文件

#include "excelHandle.h"ExcelHandle::ExcelHandle(QObject *parent) :QObject(parent) {}ExcelHandle::~ExcelHandle() {exit();delete excel;
}void ExcelHandle::init(bool showWindow, bool showAlert, QString title) {excel &#61; new QAxObject;excel -> setControl("Excel.Application");//显示窗体excel -> dynamicCall("SetVisible(bool)", showWindow);//不显示任何警告信息excel -> setProperty("DisplayAlerts", showAlert);//更改 Excel 标题栏excel -> setProperty("Caption", title);
}/*** &#64;brief ExcelHandle::save* 保存Excel*/
void ExcelHandle::save(QString url) {workbook -> dynamicCall("SaveAs(const QString&)", QDir::toNativeSeparators(url));
}/*** &#64;brief ExcelHandle::exit* 释放Excel*/
void ExcelHandle::exit() {//关闭工作簿workbooks -> dynamicCall("Close()");//退出 Excel 程序excel -> dynamicCall("Quit()");
}/*** &#64;brief ExcelHandle::openBook* 打开工作簿*/
void ExcelHandle::openBook(QString url) {workbooks &#61; excel -> querySubObject("Workbooks"); //获取工作簿集合if(QFileInfo(url).exists()) {//打开已存在的工作簿workbooks -> dynamicCall("Open(const QString&)", QString(url));}else {//添加新的工作簿workbooks -> dynamicCall("Add");}//获取当前已激活的工作簿workbook &#61; excel -> querySubObject("ActiveWorkBook");
}void ExcelHandle::openSheet(int n) {//获取当前已激活工作簿的工作表集合sheets &#61; workbook -> querySubObject("WorkSheets");//选择第 n 个工作表sheet &#61; sheets -> querySubObject("Item(int)", n);
}/*** &#64;brief ExcelHandle::setSheetName* &#64;param name* 给当前表格设定一个名称*/
void ExcelHandle::setSheetName(QString name) {sheet -> setProperty("Name", name);
}/*** &#64;brief ExcelHandle::addSheet* 增加一个Worksheet*/
void ExcelHandle::addSheet(QString name) {//获取工作表数目int sheet_count &#61; sheets -> property("Count").toInt();QAxObject *last_sheet &#61; sheets -> querySubObject("Item(int)", sheet_count);QAxObject *work_sheet &#61; sheets -> querySubObject("Add(QVariant)", last_sheet -> asVariant());last_sheet -> dynamicCall("Move(QVariant)", work_sheet -> asVariant());//设置工作表名称work_sheet -> setProperty("Name", QString("%1").arg(name));
}void ExcelHandle::changeSheet(QString name) {int count &#61; sheets -> property("Count").toInt();for(int i &#61; 1; i <&#61; count; i &#43;&#43;) {sheet &#61; sheets -> querySubObject("Item(int)", i);if(sheet -> property("Name").toString() &#61;&#61; name) { break; }}
}void ExcelHandle::changeSheet(int index) {int count &#61; sheets -> property("Count").toInt();index &#61; index > count ? count : index;sheet &#61; sheets -> querySubObject("Item(int)", index);
}/*** &#64;brief ExcelHandle::deleteSheet* 删除一个Worksheet*/
void ExcelHandle::deleteSheet(QString name) {changeSheet(name);//执行删除当前表操作sheet -> dynamicCall("delete");
}void ExcelHandle::deleteSheet(int index) {changeSheet(index);//执行删除当前表操作sheet -> dynamicCall("delete");
}void ExcelHandle::writeData(int col, int row, QVariant data) {//选中当前表格的某个单元格QAxObject *pRange &#61; sheet -> querySubObject("Cells(int,int)", col, row);//向单元格中写入数据pRange -> dynamicCall("Value", data);
}QString ExcelHandle::getEndAddr(QString start_Addr, int col, int row) {int temp_col &#61; 0, temp_row, temp_base &#61; 1;QString temp_str;QRegExp reg_exp_word("^[a-zA-Z]&#43;"),reg_exp_num("[0-9]&#43;$");//获取列数reg_exp_word.indexIn(start_Addr);temp_str &#61; reg_exp_word.cap(0);for(int i &#61; 0; i < temp_str.length(); i &#43;&#43;) {temp_col &#43;&#61; (temp_str[temp_str.length() - i - 1].unicode() - &#39;A&#39; &#43; 1) * temp_base;temp_base *&#61; 26;}temp_col &#43;&#61; col;//还原列号temp_str &#61; "";while(temp_col > 1) {temp_str &#61; QString("%1%2").arg(static_cast<char>(temp_col % 26 &#43; &#39;A&#39; - 1)).arg(temp_str);temp_col /&#61; 26;}//获取函数reg_exp_num.indexIn(start_Addr);temp_row &#61; reg_exp_num.cap(0).toInt();temp_row &#43;&#61; row;return QString("%1%2").arg(temp_str).arg(temp_row);
}void ExcelHandle::writeDataSet(QString start_Addr, QList<QList<QVariant>> data_Set) {//定义两个中转变量&#xff0c;res 用于二次列表接收&#xff0c;temp 用于一次列表接收QList<QVariant> res, temp;for(int i &#61; 0; i < data_Set.length(); i &#43;&#43;) {//清空一次接收列表temp.clear();//第一次循环将数据注入列表for(int j &#61; 0; j < data_Set[0].length(); j &#43;&#43;) {temp.append(data_Set[i][j]);}//第二次循环将数据注入列表res.append(QVariant(temp));}QString end_addr &#61; getEndAddr(start_Addr, data_Set[0].length() - 1, data_Set.length() - 1);//生成数据填充范围QString range &#61; QString("range(%1:%2)").arg(start_Addr).arg(end_addr);//确认数据填充范围QAxObject *usedRange &#61; sheet -> querySubObject(range.toUtf8().data());//根据填充范围填充数据usedRange -> setProperty("Value", QVariant(res));
}QList<QList<QVariant>> ExcelHandle::readData() {//获取被使用的表格对象usedRange &#61; sheet -> querySubObject("UsedRange");//获取被使用的表格对象中的结果集QVariant v &#61; usedRange -> dynamicCall("Value");//将结果集转换为一维列表QList<QVariant> vl &#61; v.toList();//定义一个二维列表用于存放最终结果集QList<QList<QVariant>> vll;//使用循环将一维列表转换为二维列表for(int i &#61; 0; i < vl.size(); i &#43;&#43;) {vll.append(vl[i].toList());}//返回一个二维列表return vll;
}

使用方式(该部分代码仅供参考)

QString url &#61; "E:/test11.xlsx";
QList<QVariant> vl;
QList<QList<QVariant>> vll;for(int i &#61; 0; i < 5; i &#43;&#43;) {vl.clear();for(int j &#61; 0; j < 5; j &#43;&#43;) {vl.append(i &#43; j);}vll.append(vl);
}excel_handle &#61; new ExcelHandle;
excel_handle -> init(true, false, "我是混子我怕谁");
excel_handle -> openBook(url);
excel_handle -> openSheet(1);
excel_handle -> writeData(1,2,QVariant(12));
excel_handle -> writeDataSet("C2", vll);
excel_handle -> save(url);

以上使用方式仅为参考代码&#xff0c;对于 save 函数以及 exit 函数的使用&#xff0c;一定要在文件打开的时候调用&#xff0c;不然可能会出现错误提示&#xff0c;但不会使程序崩溃&#xff0c;对于判断文件是否已被关闭&#xff0c;目前暂未研究出来&#xff0c;有研究出来的盆友可以在评论区分享一下。

源码分享&#xff1a;https://gitee.com/mjzhutianxiu/dataToExcel

学习分享&#xff0c;一起成长&#xff01;以上为小编的经验分享&#xff0c;若存在不当之处&#xff0c;请批评指正&#xff01;


推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了Python对Excel文件的读取方法,包括模块的安装和使用。通过安装xlrd、xlwt、xlutils、pyExcelerator等模块,可以实现对Excel文件的读取和处理。具体的读取方法包括打开excel文件、抓取所有sheet的名称、定位到指定的表单等。本文提供了两种定位表单的方式,并给出了相应的代码示例。 ... [详细]
  • 本文详细介绍了GetModuleFileName函数的用法,该函数可以用于获取当前模块所在的路径,方便进行文件操作和读取配置信息。文章通过示例代码和详细的解释,帮助读者理解和使用该函数。同时,还提供了相关的API函数声明和说明。 ... [详细]
  • 电话号码的字母组合解题思路和代码示例
    本文介绍了力扣题目《电话号码的字母组合》的解题思路和代码示例。通过使用哈希表和递归求解的方法,可以将给定的电话号码转换为对应的字母组合。详细的解题思路和代码示例可以帮助读者更好地理解和实现该题目。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 不同优化算法的比较分析及实验验证
    本文介绍了神经网络优化中常用的优化方法,包括学习率调整和梯度估计修正,并通过实验验证了不同优化算法的效果。实验结果表明,Adam算法在综合考虑学习率调整和梯度估计修正方面表现较好。该研究对于优化神经网络的训练过程具有指导意义。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • 怎么在PHP项目中实现一个HTTP断点续传功能发布时间:2021-01-1916:26:06来源:亿速云阅读:96作者:Le ... [详细]
  • 本文讨论了在手机移动端如何使用HTML5和JavaScript实现视频上传并压缩视频质量,或者降低手机摄像头拍摄质量的问题。作者指出HTML5和JavaScript无法直接压缩视频,只能通过将视频传送到服务器端由后端进行压缩。对于控制相机拍摄质量,只有使用JAVA编写Android客户端才能实现压缩。此外,作者还解释了在交作业时使用zip格式压缩包导致CSS文件和图片音乐丢失的原因,并提供了解决方法。最后,作者还介绍了一个用于处理图片的类,可以实现图片剪裁处理和生成缩略图的功能。 ... [详细]
  • 本文介绍了深入浅出Linux设备驱动编程的重要性,以及两种加载和删除Linux内核模块的方法。通过一个内核模块的例子,展示了模块的编译和加载过程,并讨论了模块对内核大小的控制。深入理解Linux设备驱动编程对于开发者来说非常重要。 ... [详细]
author-avatar
缪宇驰
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有