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


推荐阅读
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文详细介绍了中央电视台电影频道的节目预告,并通过专业工具分析了其加载方式,确保用户能够获取最准确的电视节目信息。 ... [详细]
  • 使用GDI的一些AIP函数我们可以轻易的绘制出简 ... [详细]
  • 本文探讨了在Java多线程环境下,如何确保具有相同key值的线程能够互斥执行并按顺序输出结果。通过优化代码结构和使用线程安全的数据结构,我们解决了线程同步问题,并实现了预期的并发行为。 ... [详细]
  • 本文将深入探讨如何在不依赖第三方库的情况下,使用 React 处理表单输入和验证。我们将介绍一种高效且灵活的方法,涵盖表单提交、输入验证及错误处理等关键功能。 ... [详细]
  • 本文详细解析了Python中的os和sys模块,介绍了它们的功能、常用方法及其在实际编程中的应用。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文探讨了 Objective-C 中的一些重要语法特性,包括 goto 语句、块(block)的使用、访问修饰符以及属性管理等。通过实例代码和详细解释,帮助开发者更好地理解和应用这些特性。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 本文探讨了将类成员属性设置为私有的重要性,并通过具体代码示例展示了如何实现对这些属性的有效控制。私有成员属性有助于增强数据的安全性和完整性,确保只有经过验证的数据才能被修改。 ... [详细]
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社区 版权所有