热门标签 | 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;


推荐阅读
  • 中科院学位论文排版指南
    随着毕业季的到来,许多即将毕业的学生开始撰写学位论文。本文介绍了使用LaTeX排版学位论文的方法,特别是针对中国科学院大学研究生学位论文撰写规范指导意见的最新要求。LaTeX以其精确的控制和美观的排版效果成为许多学者的首选。 ... [详细]
  • 2018-2019学年第六周《Java数据结构与算法》学习总结
    本文总结了2018-2019学年第六周在《Java数据结构与算法》课程中的学习内容,重点介绍了非线性数据结构——树的相关知识及其应用。 ... [详细]
  • 基于机器学习的人脸识别系统实现
    本文介绍了一种使用机器学习技术构建人脸识别系统的实践案例。通过结合Python编程语言和深度学习框架,详细展示了从数据预处理到模型训练的完整流程,并提供了代码示例。 ... [详细]
  • 历经三十年的开发,Mathematica 已成为技术计算领域的标杆,为全球的技术创新者、教育工作者、学生及其他用户提供了一个领先的计算平台。最新版本 Mathematica 12.3.1 增加了多项核心语言、数学计算、可视化和图形处理的新功能。 ... [详细]
  • 云函数与数据库API实现增删查改的对比
    本文将深入探讨使用云函数和数据库API实现数据操作(增删查改)的不同方法,通过详细的代码示例帮助读者更好地理解和掌握这些技术。文章不仅提供代码实现,还解释了每种方法的特点和适用场景。 ... [详细]
  • 在高并发需求的C++项目中,我们最初选择了JsonCpp进行JSON解析和序列化。然而,在处理大数据量时,JsonCpp频繁抛出异常,尤其是在多线程环境下问题更为突出。通过分析发现,旧版本的JsonCpp存在多线程安全性和性能瓶颈。经过评估,我们最终选择了RapidJSON作为替代方案,并实现了显著的性能提升。 ... [详细]
  • 深入解析Spring启动过程
    本文详细介绍了Spring框架的启动流程,帮助开发者理解其内部机制。通过具体示例和代码片段,解释了Bean定义、工厂类、读取器以及条件评估等关键概念,使读者能够更全面地掌握Spring的初始化过程。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • 优化SQL Server批量数据插入存储过程的实现
    本文介绍了一种改进的SQL Server存储过程,用于生成批量插入语句。该方法不仅提高了性能,还支持单行和多行模式,适用于SQL Server 2005及以上版本。 ... [详细]
  • SpringMVC RestTemplate的几种请求调用(转)
    SpringMVCRestTemplate的几种请求调用(转),Go语言社区,Golang程序员人脉社 ... [详细]
  • 本文详细介绍了优化DB2数据库性能的多种方法,涵盖统计信息更新、缓冲池调整、日志缓冲区配置、应用程序堆大小设置、排序堆参数调整、代理程序管理、锁机制优化、活动应用程序限制、页清除程序配置、I/O服务器数量设定以及编入组提交数调整等方面。通过这些技术手段,可以显著提升数据库的运行效率和响应速度。 ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 本文介绍了如何使用JavaScript的Fetch API与Express服务器进行交互,涵盖了GET、POST、PUT和DELETE请求的实现,并展示了如何处理JSON响应。 ... [详细]
  • 本文介绍了如何在 Node.js 中使用 `setDefaultEncoding` 方法为可写流设置默认编码,并提供了详细的语法说明和示例代码。 ... [详细]
  • 在PHP后端开发中遇到一个难题:通过第三方类文件发送短信功能返回的JSON字符串无法解析。本文将探讨可能的原因并提供解决方案。 ... [详细]
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社区 版权所有