QT上位机:串口调试助手
前言
上位机的简单编写可以帮我们测试并完善平台,QT作为一款跨平台的GUI开发框架,提供了非常丰富的常用串口api。本文先从最简单的串口调试助手开始,编写平台软件的串口控制界面
工程配置
QT 串口通信基于QT的QSerialPort类,先在项目文件pro中添加QT += serialport。
避免默认的widget类和ui命名,将串口部分的ui命名为ui_serial,类命名为serial
界面设置
在UI界面可以直接搭建我们想要的界面显示方案,并自动生成相应代码添加头文件,用起来还是非常方便的,注意不要和代码里手动添加的混淆就好了
需要配置的可以选择QComboBox,添加常用的配置项,比如串口号,波特率,数据校验停止位等
添加led指示灯QLabel,可以用来美化界面
点击选中的可以选择QRadioButton,用以单选,直观的给以感受
收发的界面选择文本框QPlainTextEdit,可以多行显示
手动加的短文本框可以用QLineEdit,可以单行显示
其余的用最基础的pushbutton就可以了
为了界面不会被放大缩小导致排版混乱,可以添加弹簧或者直接写死窗口大小
设计完成后,会保存为对应名称比如serial.ui

构造函数里需要将有交互的部分connect起来,将界面按键的初始值做设置
我们可以通过继承QSerialPort类,也可以直接继承Qwidget类,在里面申明Qserialport
class serial : public QWidget
{Q_OBJECTpublic:explicit serial(QWidget *parent = 0);QSerialPort *serialPort;~serial();
private slots:void serialPortReadyRead_Slot();void on_OpenBt_clicked();void on_SendBt_clicked();void on_SaveBt_clicked();void on_ClearBt_clicked();void on_pushButton_clicked();private:QPushButton *m_button;void LED(bool changeColor); //串口连接指示灯void closeEvent(QCloseEvent *event); /***关闭***/void on_comStatus(QString name, bool flag);Ui::serial *ui_serial;};
serial::serial(QWidget *parent) :QWidget(parent),ui_serial(new Ui::serial)
{ui_serial->setupUi(this);setWindowTitle("串口控制界面-HX");this->setFixedSize(980, 600); // 固定窗口的大小/***************定义接收格式数据按钮********************/QButtonGroup *btnGroupRev=new QButtonGroup(this);btnGroupRev->addButton(ui_serial->ASCII_Receive_Box,0); // 将给定的按钮添加到按钮组btnGroupRev->addButton(ui_serial->HEX_Receive_Box,1);ui_serial->ASCII_Receive_Box->setChecked(true); // 设置默认模式/***************定义发送格式数据按钮********************/QButtonGroup *btnGroupSend=new QButtonGroup(this);btnGroupSend->addButton(ui_serial->ASCII_Send_Box,0); // 将给定的按钮添加到按钮组btnGroupSend->addButton(ui_serial->HEX_Send_Box,1);ui_serial->ASCII_Send_Box->setChecked(true); // 设置默认模式serialPort = new QSerialPort(this);connect(serialPort, SIGNAL(readyRead()), this, SLOT(serialPortReadyRead_Slot()));/***************串口热插拔********************/ComChange::getInstance()->setHWND((HWND)this->winId());connect(ComChange::getInstance(), &ComChange::comStatus, this, &serial::on_comStatus);QStringList strName = ComChange::getAvailablePort(); // 获取所有可用串口ui_serial->SerialCb->addItems(strName);/*连接rtsp界面*/
// m_button = findChild<QPushButton*>("pushButton_3"); // 查找已有的pushButton3对象
// qDebug()<<"m_button is :"<<m_button;
// connect(this, &serial::buttonClicked_ctrl_left, this, &serial::on_pushButton_3_clicked);
// 连接信号与槽//}
考虑到热插拔的问题封装getAvailablePort获取当前可用串口
ComChange* ComChange::m_comChange = nullptr;
ComChange *ComChange::getInstance()
{if(m_comChange == nullptr){static QMutex mutex; //实例互斥锁。QMutexLocker locker(&mutex); //加互斥锁。if(m_comChange == nullptr){m_comChange = new ComChange();}}return m_comChange;}/*** @brief 获取系统中所有可用的串口名* @return*/
QStringList ComChange::getAvailablePort()
{QStringList strName;foreach(const QSerialPortInfo& info, QSerialPortInfo::availablePorts()){QSerialPort port(info);if(port.open(QIODevice::ReadWrite)){strName << info.portName();port.close();}}return strName;
}
串口的热插拔
/*函 数:on_comStatus描 述:串口热插拔操作。有comchange的h文件、cpp文件和该函数,才能支持串口热插拔操作,缺一不可。移植请注意。输 入:串口名称:name, 连接标志:flag输 出:无
*/
void serial::on_comStatus(QString name, bool flag)
{if(flag) ui_serial->SerialCb->addItem(name); // 串口插入时自动添加串口名else{ui_serial->SerialCb->removeItem(ui_serial->SerialCb->findText(name)); // 串口拔出时自动移除串口名LED(false); //红色LED 表示关闭串口serialPort->close(); //关闭串口ui_serial->OpenBt->setText("打开串口");}}
串口的初始化
在点击打开串口的时候做初始化,使用open(QIODevice::ReadWrite)用ReadWrite 的模式尝试打开串口,打开成功后设置串口通信的波特率,校验方式等配置。(打开方式有多种,只读(r/o)、只写(w/o)或读写(r/w)模式)
直接使用serialPort的相关API
setPortName将当前串口的名字设置为系统可用串口的名字
setBaudRate设置波特率
setDataBits设置数据位
setStopBits设置停止位
setParity设置校验位
close关闭串口
注意:串口始终以独占访问方式打开(即没有其他进程或线程可以访问已打开的串口)。
/*函 数:on_OpenBt_clicked描 述:打开串口时初始化串口输 入:无输 出:无
*/
void serial::on_OpenBt_clicked()
{if(ui_serial->OpenBt->text()=="打开串口"){serialPort->setPortName(ui_serial->SerialCb->currentText()); // 将当前串口的名字设置为系统可用串口的名字qint32 baudrate = ui_serial->BaundCb->currentText().toInt(); // 获取期望的波特率serialPort->setBaudRate(baudrate); // 设置波特率//设置数据位switch(ui_serial->DataCb->currentText().toInt()){case 8:serialPort->setDataBits(QSerialPort::Data8); break;case 7:serialPort->setDataBits(QSerialPort::Data7); break;case 6:serialPort->setDataBits(QSerialPort::Data6); break;case 5:serialPort->setDataBits(QSerialPort::Data5); break;}//设置停止位if(ui_serial->StopCb->currentText() == "1"){serialPort->setStopBits(QSerialPort::OneStop);}else if(ui_serial->StopCb->currentText() == "1.5"){serialPort->setStopBits(QSerialPort::OneAndHalfStop);}else if(ui_serial->StopCb->currentText() == "2"){serialPort->setStopBits(QSerialPort::TwoStop);}//设置校验位if(ui_serial->CheckCb->currentText() == "None"){serialPort->setParity(QSerialPort::NoParity);}else if(ui_serial->CheckCb->currentText() == "Even"){serialPort->setParity(QSerialPort::EvenParity);}else if(ui_serial->CheckCb->currentText() == "Odd"){serialPort->setParity(QSerialPort::OddParity);}LED(true); //绿色LED 表示打开串口//串口连接失败提示if(serialPort->open(QIODevice::ReadWrite) == false){QMessageBox::critical(this, "提示", "串口连接失败");LED(false); //红色LED 表示关闭串口}ui_serial->OpenBt->setText("关闭串口");}else{LED(false); //红色LED 表示关闭串口serialPort->close(); //关闭串口ui_serial->OpenBt->setText("打开串口");}
}
串口的收发
根据hex和ASCII在收发时做判断serialPort->readAll();可以接收所有信息,serialPort->write(SendTextByte); /可以通过串口将数据发送出去
/*函 数:serialPortReadyRead_Slot描 述:上位机接收数据输 入:无输 出:无
*/
void serial::serialPortReadyRead_Slot()
{QByteArray buf = serialPort->readAll(); //从串口读取信息if(ui_serial->ASCII_Receive_Box->isChecked()) //如果设置接收ASCII{ui_serial->ReceiveEdit->insertPlainText(QString::fromLocal8Bit(buf)); // 对串口接收的数据进行编码}else if(ui_serial->HEX_Receive_Box->isChecked()) //如果设置接收HEX{QDataStream out(&buf, QIODevice::ReadWrite); //读取数据while(!out.atEnd()) //读取是否完成{qint8 outChar = 0;out >> outChar;QString str = QString("%1").arg(outChar&0xFF,2,16,QLatin1Char('0')); //转换16进制ui_serial->ReceiveEdit->insertPlainText(str+" "); //每显示一次后面加一个空格}}
}/*函 数:on_SendBt_clicked描 述:上位机发送数据输 入:无输 出:无
*/
void serial::on_SendBt_clicked()
{QString sendstr=ui_serial->SendEdit->toPlainText(); //获取将要发送数据if(ui_serial->ASCII_Send_Box->isChecked()) // 如果发送ASCII模式{QByteArray SendTextByte = sendstr.toLocal8Bit(); //将发送的数据转换格式serialPort->write(SendTextByte); // 通过串口将数据发送出去}else if(ui_serial->HEX_Send_Box->isChecked()) // 如果发送HEX{QByteArray SendTextByte = QByteArray::fromHex(sendstr.toLatin1()); // 转换数据格式serialPort->write(SendTextByte); // 通过串口将数据发送出去}
}
接收区清除
为了直观的读取接收区数据,也需要增加清除当前数据的功能
/*函 数:on_ClearBt_clicked描 述:清空接收区的数据输 入:无输 出:无
*/
void serial::on_ClearBt_clicked()
{ui_serial->ReceiveEdit->clear();
}
接收数据的保存
数据量大的时候经常需要保存接收数据的log,所以存log的功能也是必不可少的
file.open(QFile::WriteOnly | QFile::Text)可以将文本数据框取出并按行排列
/*函 数:on_SaveBt_clicked描 述:以TXT格式保存接收区数据输 入:无输 出:无
*/
void serial::on_SaveBt_clicked()
{QString textFile = QFileDialog::getSaveFileName(this,tr("Save txt"),"",tr("text (*.txt)")); //选择路径//将文本框数据取出并按行排列QFile file(textFile);//文件命名if (!file.open(QFile::WriteOnly | QFile::Text)) //检测文件是否打开{QMessageBox::information(this, "Error Message", "Please Select a Text File!");return;}QTextStream out(&file); //分行写入文件out << ui_serial->ReceiveEdit->toPlainText();
}
指示灯的使用
为了美化界面,直观显示串口是否正常连接,可以加个指示灯,设置绿色表示开红色表示关
/*函 数:closeEvent描 述:关闭窗口时若未关闭串口,则要关闭串口输 入:无输 出:无
*/
void serial::closeEvent(QCloseEvent *event)
{if(serialPort->isOpen()) // 串口处于打开状态{LED(false); //红色LED 表示关闭串口serialPort->close(); // 串口关闭event->accept(); // 界面关闭}
}/*函 数:LED描 述:串口指示灯输 入:bool changeColor输 出:无
*/
void serial::LED(bool changeColor)
{if(changeColor == true){// 显示绿色ui_serial->LED->setStyleSheet("background-color: qradialgradient(spread:pad, cx:0.5, cy:0.5, radius:0.5, fx:0.5, fy:0.5, stop:0 rgba(0, 229, 0, 255), stop:1 rgba(255, 255, 255, 255));border-radius:9px;");}else{// 显示红色ui_serial->LED->setStyleSheet("background-color: qradialgradient(spread:pad, cx:0.5, cy:0.5, radius:0.5, fx:0.5, fy:0.5, stop:0 rgba(255, 0, 0, 255), stop:1 rgba(255, 255, 255, 255));border-radius:9px;");}
}
窗口退出
为了避免退出的时候占用串口,关闭窗口的时候需要关闭串口
/*函 数:closeEvent描 述:关闭窗口时若未关闭串口,则要关闭串口输 入:无输 出:无
*/
void serial::closeEvent(QCloseEvent *event)
{if(serialPort->isOpen()) // 串口处于打开状态{LED(false); //红色LED 表示关闭串口serialPort->close(); // 串口关闭event->accept(); // 界面关闭}
}
具体功能
剩下的就是根据协议规定点击pushbutton发送相应的内容了
相关文章:
QT上位机:串口调试助手
前言 上位机的简单编写可以帮我们测试并完善平台,QT作为一款跨平台的GUI开发框架,提供了非常丰富的常用串口api。本文先从最简单的串口调试助手开始,编写平台软件的串口控制界面 工程配置 QT 串口通信基于QT的QSerialPort类,先在…...
Netty核心原理与基础实战(二)——详解Bootstrap
接上篇:Netty核心原理与基础实战(一) 1 Bootstrap基础概念 Bootstrap类是Netty提供的一个便利的工厂类,可以通过它来完成Netty的客户端或服务端的Netty组件的组装,以及Netty程序的初始化和启动执行。Netty的官方解释是…...
C语言常见面试题:C语言中如何进行比较运算?
在C语言中,比较运算用于比较两个值的大小关系。比较运算符包括等于()、不等于(!)、大于(>)、小于(<)、大于等于(>)和小于等于࿰…...
学习总结14
# 【CSGRound1】天下第一 ## 题目背景 天下第一的 cbw 以主席的身份在 8102 年统治全宇宙后,开始了自己休闲的生活,并邀请自己的好友每天都来和他做游戏。由于 cbw 想要显出自己平易近人,所以 zhouwc 虽然是一个蒟蒻,也有能和 c…...
D盘不见了如何恢复?4个恢复方法(新版)!
“很奇怪!我的电脑d盘不知道为什么突然不见了,我还保存了很多重要的文件在里面呢,有什么恢复d盘的方法吗?” 在我们的日常生活中,电脑已经成为了我们工作、学习和娱乐的重要工具。然而,有时候我们会遇到一些…...
vector类的模拟实现
实现基本的vector框架 参考的是STL的一些源码,实现的vector也是看起来像是一个简略版的,但是看完能对vector这个类一些接口函数更好的认识。 我们写写成员变量,先来看看STL的成元变量是那些 namespace tjl {template<class T>class …...
Topaz Photo AI for Mac v2.3.1 补丁版人工智能降噪软件无损放大
想要将模糊的图片变得更加清晰?不妨试试Topaz Photo AI for Mac 这款人工智能、无损放大软件。Topaz Photo AI for Mac 一款强大的人工智能降噪软件,允许用户使用复杂的锐化算法来提高图像清晰度,还包括肖像编辑选项,如面部重塑、…...
【Unity3D小技巧】Unity3D中UI控制解决方案
推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址我的个人博客 大家好,我是佛系工程师☆恬静的小魔龙☆,不定时更新Unity开发技巧,觉得有用记得一键三连哦。 一、前言 在开发中总是会控制UI界面,如何优雅的控制UI界面是…...
【状态管理一】概览:状态使用、状态分类、状态具体使用
文章目录 一. 状态使用概览二. 状态的数据类型1. 算子层面2. 接口层面2.1. UML与所有状态类型介绍2.2. 内部状态:InternalKvState 将知识与实际的应用场景、设计背景关联起来,这是学以致用、刨根问底知识的一种直接方式。 本文介绍 状态数据管理&#x…...
SQL--多表查询
我们之前在讲解SQL语句的时候,讲解了DQL语句,也就是数据查询语句,但是之前讲解的查询都是单 表查询,而本章节我们要学习的则是多表查询操作,主要从以下几个方面进行讲解。 多表关系 项目开发中,在进行数据…...
多维时序 | Matlab实现CNN-RVM卷积神经网络结合相关向量机多变量时间序列预测
多维时序 | Matlab实现CNN-RVM卷积神经网络结合相关向量机多变量时间序列预测 目录 多维时序 | Matlab实现CNN-RVM卷积神经网络结合相关向量机多变量时间序列预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 Matlab实现CNN-RVM卷积神经网络结合相关向量机多变量时间序…...
RK3568平台 安卓hal3适配usb camera
一.RK安卓hal3 camera框架 Camera hal3 在 android 框架中所处的位置如上图, 对上,主要实现 Framework 一整套 API 接口,响应其 控制命令,返回数据与控制参数结果。 对下, 主要是通 V4l2 框架实现与 kernel 的交互。3a…...
使用 Visual Studio Code 在远程计算机上调试 PostgreSQL
使用 Visual Studio Code 在远程计算机上调试 PostgreSQL 1. 概述 PostgreSQL 是一个功能强大的开源关系数据库管理系统,适用于各种应用程序。在开发过程中,调试 PostgreSQL 对于识别和解决问题至关重要。在本博客中,我们将手把手教你使用客…...
javascript设计模式之建造者
工厂模式不关心过程,只关心结果,这与建造者相反,建造者更关心的是过程, 这里我们创建一个基类,其拥有技能跟爱好两个属性,还有两个实例方法用来获取技能跟爱好 // 基类 let Human function (param {}) …...
安擎科技携手华为云区块链共同打造安全天空
当前,低空经济崛起,无人机多并发、混合运行时引发的网络信息安全、空域安全问题已成行业首要课题。 在2024年1月正式实施的《民用无人驾驶航空器运行安全管理规则》(CCAR-92)第549条中规定,“无人驾驶航空器航行服务提…...
学习数据结构的第一天
结构体 如何定义结构体 1、先定义结构体类型,再定义结构体类型变量 struct student/定义学生结构体类型/ { long number; char name[20]; char sex; int age; float score[3];/三科考试成绩/ }2、定义结构体类型同时定义结构体类型变量 struct student/定义学生结…...
5.electron之主进程起一个本地服务
如果可以实现记得点赞分享,谢谢老铁~ Electron是一个使用 JavaScript、HTML 和 CSS 构建桌面应用程序的框架。 Electron 将 Chromium 和 Node.js 嵌入到了一个二进制文件中,因此它允许你仅需一个代码仓库,就可以撰写支持 Windows、…...
爬取58二手房并用SVR模型拟合
目录 一、前言 二、爬虫与数据处理 三、模型 一、前言 爬取数据仅用于练习和学习。本文运用二手房规格sepc(如3室2厅1卫)和二手房面积area预测二手房价格price,只是练习和学习,不代表任何实际意义。 二、爬虫与数据处理 import requests import cha…...
鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之RichText组件
鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之RichText组件 一、操作环境 操作系统: Windows 10 专业版、IDE:DevEco Studio 3.1、SDK:HarmonyOS 3.1 二、RichText组件 鸿蒙(HarmonyOS)富文本组件,…...
7.electron之渲染线程发送事件,主进程监听事件
如果可以实现记得点赞分享,谢谢老铁~ Electron是一个使用 JavaScript、HTML 和 CSS 构建桌面应用程序的框架。 Electron 将 Chromium 和 Node.js 嵌入到了一个二进制文件中,因此它允许你仅需一个代码仓库,就可以撰写支持 Windows、…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
Zustand 状态管理库:极简而强大的解决方案
Zustand 是一个轻量级、快速和可扩展的状态管理库,特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...
NLP学习路线图(二十三):长短期记忆网络(LSTM)
在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...
成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战
在现代战争中,电磁频谱已成为继陆、海、空、天之后的 “第五维战场”,雷达作为电磁频谱领域的关键装备,其干扰与抗干扰能力的较量,直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器,凭借数字射…...
离线语音识别方案分析
随着人工智能技术的不断发展,语音识别技术也得到了广泛的应用,从智能家居到车载系统,语音识别正在改变我们与设备的交互方式。尤其是离线语音识别,由于其在没有网络连接的情况下仍然能提供稳定、准确的语音处理能力,广…...
DeepSeek源码深度解析 × 华为仓颉语言编程精粹——从MoE架构到全场景开发生态
前言 在人工智能技术飞速发展的今天,深度学习与大模型技术已成为推动行业变革的核心驱动力,而高效、灵活的开发工具与编程语言则为技术创新提供了重要支撑。本书以两大前沿技术领域为核心,系统性地呈现了两部深度技术著作的精华:…...
MySQL的pymysql操作
本章是MySQL的最后一章,MySQL到此完结,下一站Hadoop!!! 这章很简单,完整代码在最后,详细讲解之前python课程里面也有,感兴趣的可以往前找一下 一、查询操作 我们需要打开pycharm …...
API网关Kong的鉴权与限流:高并发场景下的核心实践
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 引言 在微服务架构中,API网关承担着流量调度、安全防护和协议转换的核心职责。作为云原生时代的代表性网关,Kong凭借其插件化架构…...
C++实现分布式网络通信框架RPC(2)——rpc发布端
有了上篇文章的项目的基本知识的了解,现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...
二维FDTD算法仿真
二维FDTD算法仿真,并带完全匹配层,输入波形为高斯波、平面波 FDTD_二维/FDTD.zip , 6075 FDTD_二维/FDTD_31.m , 1029 FDTD_二维/FDTD_32.m , 2806 FDTD_二维/FDTD_33.m , 3782 FDTD_二维/FDTD_34.m , 4182 FDTD_二维/FDTD_35.m , 4793...
