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


推荐阅读
  • RTThread线程间通信
    线程中通信在裸机编程中,经常会使用全局变量进行功能间的通信,如某些功能可能由于一些操作而改变全局变量的值,另一个功能对此全局变量进行读取& ... [详细]
  • 本文将深入探讨 Unreal Engine 4 (UE4) 中的距离场技术,包括其原理、实现细节以及在渲染中的应用。距离场技术在现代游戏引擎中用于提高光照和阴影的效果,尤其是在处理复杂几何形状时。文章将结合具体代码示例,帮助读者更好地理解和应用这一技术。 ... [详细]
  • 本文深入探讨了WPF框架下的数据验证机制,包括内置验证规则的使用、自定义验证规则的实现方法、错误信息的有效展示策略以及验证时机的选择,旨在帮助开发者构建更加健壮和用户友好的应用程序。 ... [详细]
  • 入门指南:使用FastRPC技术连接Qualcomm Hexagon DSP
    本文旨在为初学者提供关于如何使用FastRPC技术连接Qualcomm Hexagon DSP的基础知识。FastRPC技术允许开发者在本地客户端实现远程调用,从而简化Hexagon DSP的开发和调试过程。 ... [详细]
  • Web动态服务器Python基本实现
    Web动态服务器Python基本实现 ... [详细]
  • 深入理解:AJAX学习指南
    本文详细探讨了AJAX的基本概念、工作原理及其在现代Web开发中的应用,旨在为初学者提供全面的学习资料。 ... [详细]
  • 高级缩放示例.就像谷歌地图一样.它仅缩放图块,但不缩放整个图像.因此,缩放的瓷砖占据了恒定的记忆,并且不会为大型缩放图像调整大小的图像.对于简化的缩放示例lookhere.在Win ... [详细]
  • 在Qt框架中,信号与槽机制是一种独特的组件间通信方式。本文探讨了这一机制相较于传统的C风格回调函数所具有的优势,并分析了其潜在的不足之处。 ... [详细]
  • 原文地址:https:blog.csdn.netqq_35361471articledetails84715491原文地址:https:blog.cs ... [详细]
  • 文章目录前言Program(程序)Identifier(标识符)Literal(字面量)Vari ... [详细]
  • 在Effective Java第三版中,建议在方法返回类型中优先考虑使用Collection而非Stream,以提高代码的灵活性和兼容性。 ... [详细]
  • 本文介绍了实时流协议(RTSP)的基本概念、组成部分及其与RTCP的交互过程,详细解析了客户端请求格式、服务器响应格式、常用方法分类及协议流程,并提供了SDP格式的深入解析。 ... [详细]
  • 本文探讨了在UIScrollView上嵌入Webview时遇到的一个常见问题:点击图片放大并返回后,Webview无法立即滑动。我们将分析问题原因,并提供有效的解决方案。 ... [详细]
  • 本文介绍了如何使用Java实现数组的冒泡排序算法,以及如何利用Scanner类动态给数组赋值并进行数组扩容。文章详细解释了冒泡排序的原理和步骤,并提供了代码示例。 ... [详细]
  • 本文介绍了如何在 VC++ 中使用双缓冲技术来减少屏幕闪烁,提高绘图性能。通过示例代码详细展示了双缓冲的实现步骤。 ... [详细]
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社区 版权所有