干货|使用Qt将采集到的结果数据保存至Excel中

原创 嵌入式从0到1 2021-07-12 23:17

大家好,我是小哈哥,今天我们接着上一篇网文的内容,继续完成网友问答

今天我们来分享第三个问题的解决方案:

  • 基于Modbus协议将电压数据上传至上位机
  • Qt程序解析Modbus协议,并将解析之后的结果显示在曲线中
  • 将声音强度数据保存至Excel中

本次分享的实例在 分享一个非常强大且好用的绘图控件QCustomPlot 的程序基础上完成。

要解决这个问题,我们要先了解一下,什么是VBA

VBA

很多人听过大名鼎鼎的VBA,至于怎么用可能不太熟悉,那么VBA是什么呢?

VBA(Visual Basic for Applications)是Visual Basic的一种宏语言,是在其桌面应用程序中执行通用的自动化(OLE)任务的编程语言。主要能用来扩展Windows的应用程序功能,特别是Microsoft Office软件。它也可说是一种应用程式视觉化的 Basic 脚本。

Office取得巨大成功的一个重要原因就是VBA,使用VBA可以完成很多事情,基于Excel、Word的VBA小程序不计其数。掌握对VBA语言的使用,可以让复杂的工作简易化,减少不必要的重复性工作,大大提高我们的工作效率。

VBA 是基于Visual Basic 发展而来的,它们具有相似的语言结构。

我们来一个VBA小程序

VBA是依附于Office的。

在新工作表的Sheet1上点击右键,弹出菜单中选择“查看代码”。

或者直接使用快捷键 Alt+F11 ,打开VBA编辑器:

在编辑框中写入如下代码:

Sub TestDemo()
    MsgBox ("Hello world")
End Sub

我们点击运行按钮,可以查看此过程的运行结果。

这样我们的第一个VBA程序就完成了。

将上面的过程改成函数,我们还可以在单元格的双击事件中调用该函数,具体代码如下:

Function TestDemo()
    MsgBox ("Hello world")
End Function

Private Sub Worksheet_BeforeDoubleClick(ByVal Target As Range, Cancel As Boolean)
    ' Call TestDemo
    Application.Run TestDemo()
End Sub

这样,我们在Excel中,双击单元格,就可以弹出内容为Hello world的对话框了,具体演示效果如下:

VBA中的过程(Procedure)有两种,它们都是一个可以获取参数、执行一系列语句、以及改变其参数的值的独立过程。

一种叫函数(Function)           '可调用代码block 

一种叫子程序(Subroutine)    '可执行代码block

VBA操作Excel单元格

目标:生成一个Excel,然后修改Sheet的名字并让单元格内填充内容,具体演示效果如下:

涉及的完整代码如下:

Sub AddWorkbooks()
    Set wb 
= Workbooks.Add
    Set ws = wb.Sheets(1)
    ws.Name = "嵌入式从0到1"
    ws.Cells(11) = "序号"
    ws.Cells(12) = "声音强度"
    For i = 2 To 10
        ws.Cells(i, 1) = i - 1
    Next i
    ActiveWorkbook.SaveAs Filename:=ThisWorkbook.Path & Application.PathSeparator & "test.xlsx", FileFormat:=xlOpenXMLWorkbook
    ActiveWindow.Close
End Sub

.xlsx文件为什么不能保存VBA代码呢?

2003版(.xls文件)可以保存带VBA代码的文件,不用更改扩展名。但是,当你要运行宏时,必须在安全性设置里启用宏,比较麻烦。2007版以上进行了改进,可以直接把带VBA代码的文件另存为.xlsm文件。这样,打开这个文件,可以直接运行宏。

.xlsx文件保存带有VBA代码的文件会有提示,将此文件另存为.xlsm文件即可避免提示。

.xlsm文件是“启用宏的工作簿”,也就是含有VBA代码的表格。

Qt的Excel操作常用类

在VBA的参考手册中就可以看到具体函数、属性的用法,Qt操作Excel主要通过 QAxObject + Excel VBA来实现!

QAxObject *excel = NULL;
excel = new QAxObject("Excel.Application");
  1. 设置当前进程界面不可见:
excel->dynamicCall("SetVisible(bool)"false);
  1. 获取工作簿集合:
QAxObject *workbooks = NULL;
workbooks = excel->querySubObject("WorkBooks");
  1. 打开已存在的工作簿:
QAxObject *workbook = NULL;
QString templateFilePath = qApp->applicationDirPath() + "/template.xlsx";
workbook = workbooks->querySubObject("Open(QString&)",templateFilePath);
  1. 获取所有的工作表:
QAxObject * worksheets = workbook->querySubObject("WorkSheets");
  1. 获取第一个工作表:
QAxObject * worksheet = worksheets->querySubObject("Item(int)"1);
  1. 获取表范围:
QAxObject * usedrange = worksheet->querySubObject("UsedRange");
  1. 获取行数:
QAxObject * rows = usedrange->querySubObject("Rows");    
int iRows = rows->property("Count").toInt();
//获取起始行
int iStartRow = rows->property("Row").toInt();
  1. 获取列数:
QAxObject * columns = usedrange->querySubObject("Columns");    
int iColumns = columns->property("Count").toInt();、
//获取起始列
int iColumn = columns->property("Column").toInt();
  1. 设置某单元格的值:
QString cora = "A" + QString::number(iStartRow+iRows+i);        
QAxObject *rangea = worksheet->querySubObject("Range(QString)", cora);        
rangea->setProperty("Value", arrX[i]);

将声音强度保存至Excel中

首先需要在pro文件中添加axcontainer

QT += core gui printsupport axcontainer
QT += serialport

然后添加包含文件

#include <QAxObject>

如下是代码的具体实现:

void MainWindow::on_btnExportData_clicked()
{
    QAxObject *excel = NULL;
    QAxObject *workbooks = NULL;
    QAxObject *workbook = NULL;
    excel = new QAxObject("Excel.Application");
    excel->dynamicCall("SetVisible(bool)"false);
    workbooks = excel->querySubObject("WorkBooks");

    QString templateFilePath = qApp->applicationDirPath() + "/template.xlsx";

    workbook = workbooks->querySubObject("Open(QString&)",templateFilePath);

    //获取活动工作簿
    QAxObject * worksheets = workbook->querySubObject("WorkSheets");        //获取所有的工作表

    QAxObject * worksheet = worksheets->querySubObject("Item(int)"1);

    //表格范围和行数列数
    QAxObject * usedrange = worksheet->querySubObject("UsedRange");

    QAxObject * rows = usedrange->querySubObject("Rows");
    int iRows = rows->property("Count").toInt();

    QAxObject * columns = usedrange->querySubObject("Columns");
    int iColumns = columns->property("Count").toInt();

    int iStartRow = rows->property("Row").toInt();

    int iColumn = columns->property("Column").toInt();

    //写入数据
    for(int i = 0;i<arrY.size();i++)
    {
        QString cora = "A" + QString::number(iStartRow+iRows+i);
        QAxObject *rangea = worksheet->querySubObject("Range(QString)", cora);
        rangea->setProperty("Value", arrX[i]);

        QString corb = "B" + QString::number(iStartRow+iRows+i);
        QAxObject *rangeb = worksheet->querySubObject("Range(QString)", corb);
        rangeb->setProperty("Value", arrY[i]);
    }

    workbook->dynamicCall("Save()");   //!保存文件
    workbook->dynamicCall("Close()");
    excel->dynamicCall("Quit()");

    if (excel)
    {
        delete excel;
        excel = NULL;
    }

    QMessageBox::information(0 ,"提示" ,"数据保存成功", QMessageBox::Ok | QMessageBox::Default , 0 );
}

结果展示

本实例实现的功能是打开一个模板文件,把传入的数据写到指定的单元格,并自动完成保存和关闭的操作。

下次导出的数据会追加到上次导出数据的尾部。

具体演示效果如下:

对于简单的数据保存,可以使用 .csv 文件,操作简单。如果想要使用Excel的复杂功能,可以使用本文的方法进行操作。

欢迎关注

关注公众号:嵌入式从0到1,第一时间获取技术干货,玩模块、学硬件,带你从0走到1,欢迎关注!

公众号内容包括但不限于STM32、单片机、物联网、鸿蒙、Qt、小程序,欢迎感兴趣的朋友,持续关注。

想加技术交流群的,可以加我微信:chengxuyuanxiaoha,让我拉你进群,与更多同道中人一起成长。

参考阅读

上位机如何获得环境声音强度的大小

分享一个非常强大且好用的绘图控件QCustomPlot




点个在看你最好看




嵌入式从0到1 专注于嵌入式知识分享
评论
  • ALINX 正式发布 AMD Virtex UltraScale+ 系列 FPGA PCIe 3.0 综合开发平台 AXVU13P!这款搭载 AMD 16nm 工艺 XCVU13P 芯片的高性能开发验证平台,凭借卓越的计算能力和灵活的扩展性,专为应对复杂应用场景和高带宽需求而设计,助力技术开发者加速产品创新与部署。随着 5G、人工智能和高性能计算等领域的迅猛发展,各行业对计算能力、灵活性和高速数据传输的需求持续攀升。FPGA 凭借其高度可编程性和实时并行处理能力,已成为解决行业痛点的关
    ALINX 2024-12-20 17:44 73浏览
  • Supernode与艾迈斯欧司朗携手,通过Belago红外LED实现精准扫地机器人避障;得益于Belago出色的红外补光功能,使扫地机器人能够大大提升其识别物体的能力,实现精准避障;Belago点阵照明器采用迷你封装,兼容标准无铅回流工艺,适用于各种3D传感平台,包括移动设备、物联网设备和机器人。全球领先的光学解决方案供应商艾迈斯欧司朗(瑞士证券交易所股票代码:AMS)近日宣布,与国内领先的多行业三维视觉方案提供商超节点创新科技(Supernode)双方联合推出采用艾迈斯欧司朗先进Belago红
    艾迈斯欧司朗 2024-12-20 18:55 70浏览
  • 百佳泰特为您整理2024年12月各大Logo的最新规格信息。——————————USB▶ 百佳泰获授权进行 USB Active Cable 认证。▶ 所有符合 USB PD 3.2 标准的产品都有资格获得USB-IF 认证——————————Bluetooth®▶ Remote UPF Testing针对所有低功耗音频(LE Audio)和网格(Mesh)规范的远程互操作性测试已开放,蓝牙会员可使用该测试,这是随时测试产品的又一绝佳途径。——————————PCI Express▶ 2025年
    百佳泰测试实验室 2024-12-20 10:33 114浏览
  •         不卖关子先说感受,真本书真是相见恨晚啊。字面意思,见到太晚了,我刚毕业或者刚做电子行业就应该接触到这本书的。我自己跌跌撞撞那么多年走了多少弯路,掉过多少坑,都是血泪史啊,要是提前能看到这本书很多弯路很多坑都是可以避免的,可惜这本书是今年出的,羡慕现在的年轻人能有这么丰富完善的资料可以学习,想当年我纯靠百度和论坛搜索、求助啊,连个正经师傅都没有,从软件安装到一步一布操作纯靠自己瞎摸索,然后就是搜索各种教程视频,说出来都是泪啊。  &
    DrouSherry 2024-12-19 20:00 109浏览
  • 随着工业自动化和智能化的发展,电机控制系统正向更高精度、更快响应和更高稳定性的方向发展。高速光耦作为一种电气隔离与信号传输的核心器件,在现代电机控制中扮演着至关重要的角色。本文将详细介绍高速光耦在电机控制中的应用优势及其在实际工控系统中的重要性。高速光耦的基本原理及优势高速光耦是一种光电耦合器件,通过光信号传递电信号,实现输入输出端的电气隔离。这种隔离可以有效保护电路免受高压、电流浪涌等干扰。相比传统的光耦,高速光耦具备更快的响应速度,通常可以达到几百纳秒到几微秒级别的传输延迟。电气隔离:高速光
    晶台光耦 2024-12-20 10:18 139浏览
  • 汽车驾驶员监控系统又称DMS,是一种集中在车辆中的技术,用于实时跟踪和评估驾驶员状态及驾驶行为。随着汽车产业智能化转型,整合AI技术的DMS逐渐成为主流,AI模型通过大量数据进行持续训练,使得驾驶监控更加高效和精准。 驾驶员监测系统主要通过传感器、摄像头收集驾驶员的面部图像,定位头部姿势、人脸特征及行为特征,并通过各种异常驾驶行为检测模型运算来识别驾驶员的当前状态。如果出现任何异常驾驶行为(如疲劳,分心,抽烟,接打电话,无安全带等),将发出声音及视觉警报。此外,驾驶员的行为数据会被记录
    启扬ARM嵌入式 2024-12-20 09:14 94浏览
  • 国产数字隔离器已成为现代电子产品中的关键部件,以增强的性能和可靠性取代了传统的光耦合器。这些隔离器广泛应用于医疗设备、汽车电子、工业自动化和其他需要强大信号隔离的领域。准确测试这些设备是确保其质量和性能的基本步骤。如何测试数字隔离器测试数字隔离器需要精度和正确的工具集来评估其在各种条件下的功能和性能。以下设备对于这项任务至关重要:示波器:用于可视化信号波形并测量时序特性,如传播延迟、上升时间和下降时间。允许验证输入输出信号的完整性。频谱分析仪:测量电磁干扰(EMI)和其他频域特性。有助于识别信号
    克里雅半导体科技 2024-12-20 16:35 59浏览
  • //```c #include "..\..\comm\AI8051U.h"  // 包含头文件,定义了硬件寄存器和常量 #include "stdio.h"              // 标准输入输出库 #include "intrins.h"         &n
    丙丁先生 2024-12-20 10:18 84浏览
  • 汽车行业的变革正愈演愈烈,由交通工具到“第三生活空间”。业内逐渐凝聚共识:汽车的下半场在于智能化。而智能化的核心在于集成先进的传感器,以实现高等级的智能驾驶乃至自动驾驶,以及更个性、舒适、交互体验更优的智能座舱。毕马威中国《聚焦电动化下半场 智能座舱白皮书》数据指出,2026年中国智能座舱市场规模将达到2127亿元,5年复合增长率超过17%。2022年到2026年,智能座舱渗透率将从59%上升至82%。近日,在SENSOR CHINA与琻捷电子联合举办的“汽车传感系列交流会-智能传感专场”上,艾
    艾迈斯欧司朗 2024-12-20 19:45 77浏览
  • 光耦固态继电器(SSR)作为现代电子控制系统中不可或缺的关键组件,正逐步取代传统机械继电器。通过利用光耦合技术,SSR不仅能够提供更高的可靠性,还能适应更加复杂和严苛的应用环境。在本文中,我们将深入探讨光耦固态继电器的工作原理、优势、挑战以及未来发展趋势。光耦固态继电器:如何工作并打破传统继电器的局限?光耦固态继电器通过光电隔离技术,实现输入信号与负载之间的电气隔离。其工作原理包括三个关键步骤:光激活:LED接收输入电流并发出与其成比例的光信号。光传输:光电传感器(如光电二极管或光电晶体管)接收
    腾恩科技-彭工 2024-12-20 16:30 46浏览
  • 光耦合器,也称为光隔离器,是用于电气隔离和信号传输的多功能组件。其应用之一是测量电路中的电压。本文介绍了如何利用光耦合器进行电压测量,阐明了其操作和实际用途。使用光耦合器进行电压测量的工作原理使用光耦合器进行电压测量依赖于其在通过光传输信号的同时隔离输入和输出电路的能力。该过程包括:连接到电压源光耦合器连接在电压源上。输入电压施加到光耦合器的LED,LED发出的光与施加的电压成比例。光电二极管响应LED发出的光由输出侧的光电二极管或光电晶体管检测。随着LED亮度的变化,光电二极管的电阻相应减小,
    腾恩科技-彭工 2024-12-20 16:31 59浏览
  • 耳机虽看似一个简单的设备,但不仅只是听音乐功能,它已经成为日常生活和专业领域中不可或缺的一部分。从个人娱乐到专业录音,再到公共和私人通讯,耳机的使用无处不在。使用高质量的耳机不仅可以提供优良的声音体验,还能在长时间使用中保护使用者听力健康。耳机产品的质量,除了验证产品是否符合法规标准,也能透过全面性的测试和认证过程,确保耳机在各方面:从音质到耐用性,再到用户舒适度,都能达到或超越行业标准。这不仅保护了消费者的投资,也提升了该公司在整个行业的产品质量和信誉!客户面临到的各种困难一家耳机制造商想要透
    百佳泰测试实验室 2024-12-20 10:37 149浏览
我要评论
1
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦