news 2026/6/15 22:07:58

Qt 调用 DLL 实现固件升级进度弹窗(完整实战案例)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Qt 调用 DLL 实现固件升级进度弹窗(完整实战案例)

在工业软件、设备配置工具或上位机系统中,**固件升级(Firmware Upgrade)**是一个非常常见但又容易出问题的功能点。
典型需求包括:

  • 调用厂商提供的 Windows DLL 执行升级
  • 升级过程不能阻塞 UI
  • 实时显示升级进度
  • 升级过程中禁止用户误操作
  • 升级完成后给出明确反馈

本文通过一个真实工程级示例,完整讲解如何使用Qt + Windows DLL实现一个可靠的固件升级进度弹窗


一、应用场景说明(为什么要这样做)

在实际项目中,固件升级逻辑通常由硬件厂商提供的 DLL完成,例如:

  • LVDS / V-by-One 显示控制器
  • FPGA / MCU 配置工具
  • 工控采集设备
  • 显示驱动板卡升级程序

Qt 作为上位机 UI 层,只负责:

  • 加载 DLL
  • 调用升级接口
  • 显示升级状态
  • 控制用户交互

升级过程本身必须在后台线程执行,否则 UI 会卡死。


二、设计思路概览

整体架构如下:

  1. 使用LoadLibrary / GetProcAddress动态加载 DLL
  2. 在后台线程中调用升级函数
  3. 通过定时器轮询 DLL 中的升级进度接口
  4. 使用QDialog + QProgressBar显示进度
  5. 升级完成后释放 DLL 并允许用户关闭窗口

这是工业软件中最常见、最稳定的一种升级实现方式


三、DLL 接口定义说明

假设厂商 DLL 中提供如下接口:

// 启动固件升级voidUploadProgram(constchar*filePath);// 查询升级进度(0 ~ 100)intGetUploadProgramSchedule();

Qt 端通过函数指针方式调用。


四、核心实现代码(完整案例)

1. 函数整体结构

voidLVDSVbyOneSignalCollector::firmwareUpInformationBox(QString path)

该函数的职责是:

  • 弹出升级对话框
  • 启动固件升级
  • 显示升级进度
  • 等待升级完成

2. 动态加载 DLL

HMODULE hDll=LoadLibraryA("LVDSVbyOneSignalCollectorDLL.dll");if(!hDll){return;}autopUploadProgram=(PFN_UploadProgram)GetProcAddress(hDll,"UploadProgram");autopGetUploadProgramSchedule=(PFN_GetUploadProgramSchedule)GetProcAddress(hDll,"GetUploadProgramSchedule");if(!pUploadProgram||!pGetUploadProgramSchedule){FreeLibrary(hDll);return;}

为什么使用动态加载?

  • DLL 由第三方提供
  • 可选功能模块
  • 避免启动时强依赖
  • 方便版本替换

3. 构建升级 UI(不可中断)

QDialog*dialog=newQDialog(nullptr);dialog->setWindowTitle(tr("Firmware Upgrade"));dialog->setWindowModality(Qt::ApplicationModal);dialog->setFixedSize(360,140);// 禁止右上角关闭dialog->setWindowFlags(dialog->windowFlags()&~Qt::WindowCloseButtonHint);

这是一个应用级模态窗口,升级过程中用户无法操作主界面。


4. UI 元素布局

QLabel*label=newQLabel(tr("Upgrading firmware, please wait..."),dialog);QProgressBar*progressBar=newQProgressBar(dialog);progressBar->setRange(0,100);progressBar->setValue(0);QPushButton*btnOk=newQPushButton(tr("OK"),dialog);btnOk->setEnabled(false);

升级未完成前,OK 按钮不可点击,避免误关闭。


5. 后台线程执行升级(关键)

QFuture<void>future=QtConcurrent::run([=](){pUploadProgram(rInfor.filePath);});

这里使用QtConcurrent::run的目的非常明确:

  • 避免 UI 阻塞
  • 升级逻辑与界面解耦
  • 不需要手写 QThread

这是 Qt 工程中推荐的做法之一


6. 使用 QTimer 轮询升级进度

QTimer*timer=newQTimer(dialog);QObject::connect(timer,&QTimer::timeout,dialog,[=](){intvalue=pGetUploadProgramSchedule();// 0~100progressBar->setValue(value);if(future.isFinished()){timer->stop();progressBar->setValue(100);label->setText(tr("Firmware upgrade completed."));btnOk->setEnabled(true);FreeLibrary(hDll);}});timer->start(300);

这种设计的优点:

  • 不需要 DLL 回调
  • 逻辑简单、稳定
  • 工业项目中极其常见

7. 升级完成后退出

QObject::connect(btnOk,&QPushButton::clicked,dialog,[=](){dialog->accept();dialog->deleteLater();});

只有在升级完成后,用户才能关闭窗口。


五、工程级注意事项(非常重要)

1️⃣ 文件路径生命周期问题

std::string str=path.toStdString();rInfor.filePath=str.c_str();

如果 DLL 在内部异步使用该指针,这里存在风险。
更安全的方式是保证字符串长期有效(如成员变量或深拷贝)。


2️⃣ DLL 卸载时机

FreeLibrary必须确保:

  • 升级线程完全结束
  • DLL 内部没有残留线程

否则可能导致随机崩溃。


3️⃣ 建议增加失败与超时处理

生产环境中建议补充:

  • 升级失败状态
  • 超时检测
  • 错误码提示
  • 日志记录

六、总结

本文通过一个真实工业项目中的固件升级案例,展示了:

  • Qt 如何调用 Windows DLL
  • 如何在后台线程执行耗时任务
  • 如何安全地显示升级进度
  • 如何设计不可中断的升级 UI

这种结构在工业上位机 / 设备工具 / 显示控制软件中非常通用,具有很高的工程参考价值。


voidLVDSVbyOneSignalCollector::firmwareUp(){QMessageBox msgBox;msgBox.setWindowTitle("提示");msgBox.setText("你确定要继续吗?");msgBox.setIcon(QMessageBox::Question);msgBox.setStandardButtons(QMessageBox::Ok|QMessageBox::Cancel);msgBox.button(QMessageBox::Ok)->setText("更新");msgBox.button(QMessageBox::Cancel)->setText("取消");intret=msgBox.exec();if(ret==QMessageBox::Ok){qDebug()<<"用户点击了确定按钮";QString path=ui.lineEdit_bootPath->text();QFileInfofileInfo(path);if(fileInfo.exists()){if(fileInfo.isDir()){qDebug()<<"路径是有效的目录";}elseif(fileInfo.isFile()){qDebug()<<"路径是有效的文件";}}else{qDebug()<<"路径无效";QMessageBox::information(this,"path",QString("The path is invalid."));return;}firmwareUpInformationBox(path);}elseif(ret==QMessageBox::Cancel){qDebug()<<"用户点击了关闭按钮";}}voidLVDSVbyOneSignalCollector::firmwareUpInformationBox(QString path){HMODULE hDll=LoadLibraryA("LVDSVbyOneSignalCollectorDLL.dll");if(!hDll){return;}PFN_UploadProgram pUploadProgram=(PFN_UploadProgram)GetProcAddress(hDll,"UploadProgram");PFN_GetUploadProgramSchedule pGetUploadProgramSchedule=(PFN_GetUploadProgramSchedule)GetProcAddress(hDll,"GetUploadProgramSchedule");if(!pUploadProgram||!pGetUploadProgramSchedule){FreeLibrary(hDll);return;}std::string str=path.toStdString();rInfor.filePath=str.c_str();/* ================== UI ================== */QDialog*dialog=newQDialog(nullptr);dialog->setWindowTitle(tr("Firmware Upgrade"));dialog->setWindowModality(Qt::ApplicationModal);dialog->setFixedSize(360,140);// 禁止关闭dialog->setWindowFlags(dialog->windowFlags()&~Qt::WindowCloseButtonHint);QLabel*label=newQLabel(tr("Upgrading firmware, please wait..."),dialog);QProgressBar*progressBar=newQProgressBar(dialog);progressBar->setRange(0,100);progressBar->setValue(0);QPushButton*btnOk=newQPushButton(tr("OK"),dialog);btnOk->setEnabled(false);// 完成前不可点击QVBoxLayout*layout=newQVBoxLayout(dialog);layout->addWidget(label);layout->addWidget(progressBar);layout->addWidget(btnOk,0,Qt::AlignRight);dialog->setLayout(layout);dialog->show();/* ================== 后台线程 ================== */QFuture<void>future=QtConcurrent::run([=](){pUploadProgram(rInfor.filePath);});/* ================== 进度轮询 ================== */QTimer*timer=newQTimer(dialog);QObject::connect(timer,&QTimer::timeout,dialog,[=](){intvalue=pGetUploadProgramSchedule();// 0~100progressBar->setValue(value);if(future.isFinished()){timer->stop();progressBar->setValue(100);label->setText(tr("Firmware upgrade completed."));btnOk->setEnabled(true);FreeLibrary(hDll);}});timer->start(300);/* ================== 完成后退出 ================== */QObject::connect(btnOk,&QPushButton::clicked,dialog,[=](){dialog->accept();dialog->deleteLater();});}
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 13:19:26

Open-AutoGLM无法接入微信?5大核心技术瓶颈全解析

第一章&#xff1a;Open-AutoGLM无法接入微信&#xff1f;5大核心技术瓶颈全解析在将 Open-AutoGLM 接入微信生态的过程中&#xff0c;开发者普遍遭遇连接失败、消息延迟或认证超时等问题。这些问题背后&#xff0c;往往源于五大核心技术瓶颈&#xff0c;涉及协议兼容性、身份验…

作者头像 李华
网站建设 2026/6/15 19:35:18

【AI工程化核心突破】:cogagent如何驱动Open-AutoGLM实现端到端AutoGLM?

第一章&#xff1a;AI工程化背景下的AutoGLM演进随着人工智能技术从实验室研究逐步走向工业级应用&#xff0c;AI工程化已成为推动大模型落地的核心驱动力。在这一背景下&#xff0c;AutoGLM作为面向通用语言模型的自动化训练与推理框架&#xff0c;经历了从实验原型到生产就绪…

作者头像 李华
网站建设 2026/6/15 14:19:25

【AI新时代来临】:Open-AutoGLM或将改写ChatGPT主导的全球格局

第一章&#xff1a;Open-AutoGLM会和chatgpt一样火 随着大语言模型技术的不断演进&#xff0c;开源生态正迎来新一轮爆发。Open-AutoGLM作为一款聚焦自动化生成与推理能力的开源模型&#xff0c;凭借其高效的上下文理解机制和模块化架构设计&#xff0c;正在吸引越来越多开发者…

作者头像 李华
网站建设 2026/6/15 12:40:45

HubSpot 入门指南(3):什么是 Sales Hub?热门功能讲解与定价说明

优阅达致力于与 HubSpot 一道&#xff0c;为全球客户提供更多高效、智能和创新的营销自动化解决方案。 作为 HubSpot 亚太区铂金级合作伙伴与授权代理商&#xff0c;优阅达&#xff08;DKMECO&#xff09;在服务上千家中国本土及出海企业的过程中&#xff0c;发现&#xff1a; …

作者头像 李华
网站建设 2026/6/15 19:32:14

小米AI女神罗福莉大模型全解析:从MiMo到具身智能的技术突破

小米大模型技术负责人罗福莉提出"大模型解码人类思维投影"理论&#xff0c;主导开发MiMo系列模型。其中MiMo-V2-Flash通过MoE稀疏架构、混合注意力、多词元预测和MOPD蒸馏等技术&#xff0c;实现效率与性能的突破性平衡。她还推动模型向多模态和具身智能发展&#xf…

作者头像 李华
网站建设 2026/6/15 13:56:25

DeepSeek大模型微调实战:关键参数与优化策略

DeepSeek大模型虽具备通用知识&#xff0c;但在特定场景下需通过微调提升性能。微调能显著增强模型在医疗、金融等垂直领域的专业能力&#xff0c;满足个性化需求。微调过程包括环境搭建、数据处理、模型配置和训练执行等关键步骤&#xff0c;需合理设置LoRA参数、学习率、批次…

作者头像 李华