QTDemo:串口调试工具

server/2024/12/29 0:22:55/

项目简介

本项目通过QT框架设计一款可以在Windows、Linux等平台的跨平台串口助手,串口功能能够满足基本的调试需求。

本项目采用的版本为:QT5.14 + visual studio 2022 进行开发。

项目源码:https://github.com/say-Hai/MyCOMDemo

项目页面:

image-20241220120411588

一、创建开发环境

打开vs新建工程,选择创建Qt Widgets Application项目,选择保存路径后,配置QT的SerialPort模块。

image-20241217214638298

二、配置ui界面

打开工程的ui文件,设置本项目的ui页面(可直接从本项目的ui文件中copy到自己的项目中;但是注意:需要暂时把comboBoxNo_2降级成普通QComboBox

image-20241218103406284

image-20241218102410011

三、编写串口扫描代码

通过QSerialPortInfo::availablePorts生成可用串口列表,(目前暂定在MyCOM.h的构造函数中编写串口列表函数)

MyCOM::MyCOM(QWidget* parent): QMainWindow(parent)
{ui.setupUi(this);//创建串口列表QStringList comPort;foreach(const QSerialPortInfo & info, QSerialPortInfo::availablePorts()){comPort << info.portName();}ui.comboBoxNo_2->addItems(comPort);
}

image-20241218103454075

四、“打开串口”按钮设计

vs中无法使用Qt Creator的“转到槽”功能,因此需要开发者自己绑定槽函数;具体操作步骤为:https://www.cnblogs.com/ybqjymy/p/17999513

注:解决vs + qt 导致的乱码问题:出现中文的文件首行加上#pragma execution_character_set("utf-8")

当我们绑定好槽函数on_pushButtonOpen_clicked(),接下来就是实现串口打开逻辑:以下为具体代码

//Map定义代码查看源文件
void MyCOM::on_pushButtonOpen_clicked()
{QSerialPort::BaudRate CombaudRate;QSerialPort::DataBits ComdataBits;QSerialPort::StopBits ComstopBits;QSerialPort::Parity   ComParity;QString selectedBaudRate = ui.comboBoxComBaud_2->currentText();std::cout << selectedBaudRate.toStdString() << "\n";if (baudRateMap.contains(selectedBaudRate)) {CombaudRate = baudRateMap[selectedBaudRate];}else {// 如果用户选择了一个未知的波特率,可以设置默认值或提示错误CombaudRate = QSerialPort::Baud9600; // 默认值qWarning("Invalid baud rate selected. Defaulting to 9600.");}
//具体代码查看源文件// 根据用户选择设置数据位// 根据用户选择设置停止位// 根据用户选择设置校验方式//初始化串口MyCom.setBaudRate(CombaudRate);MyCom.setDataBits(ComdataBits);MyCom.setStopBits(ComstopBits);MyCom.setParity(ComParity);MyCom.setPortName(spTxt);//打开串口if (ui.pushButtonOpen_2->text() == "打开串口"){bool ComFlag;ComFlag = MyCom.open(QIODevice::ReadWrite);if (ComFlag == true)//串口打开成功{//串口下拉框设置为不可选ui.comboBoxCheck_2->setEnabled(false);//具体代码查看源文件//使能相应按钮等ui.pushButtonSend_2->setEnabled(true);//具体代码查看源文件ui.pushButtonOpen_2->setText(" 关闭串口 ");}else{QMessageBox::critical(this, "错误提示", "串口打开失败,该端口可能被占用或不存在!rnLinux系统可能为当前用户无串口访问权限!");}}else{MyCom.close();ui.pushButtonOpen_2->setText(" 打开串口 ");//具体代码查看源文件//使相应的按钮不可用ui.pushButtonSend_2->setEnabled(false);具体代码查看源文件}
}

五、串口数据发送与接收

通过信号槽机制,在发送区发送数据,通过&QIODevice::readyRead信号来通知接收区函数&MyCOM::MyComRevSlot打印串口发送的数据

代码逻辑:

  • 信号槽逻辑:当串口有数据可以读取时,自动响应MyComRevSlot函数。

    connect(&MyCom, &QIODevice::readyRead, this, &MyCOM::MyComRevSlot);
    
  • 发送区代码逻辑:通过第四步中的“转到槽”机制,在发送按钮上绑定槽函数on_pushButtonSend_clicked(),再槽函数中接收发送区字符并通过MyCom.write(comSendData)发送到串口。

    • 其中16进制发送需要将字符串格式化成16进制 QByteArray::fromHex(SendTemp.toUtf8()).data();
    //精简版,少了一些单选框的逻辑判断
    void MyCOM::on_pushButtonSend_clicked()
    {QByteArray comSendData;QString SendTemp;int temp;//读取发送窗口数据SendTemp = ui.TextSend_2->toPlainText();//判断发送格式,并格式化数据if (ui.checkBoxSendHex_2->checkState() != false)//16进制发送{comSendData = QByteArray::fromHex(SendTemp.toUtf8()).data();//获取字符串}temp = MyCom.write(comSendData);
    }
    
  • 接收区代码逻辑:通过信号槽机制来调用MyComRevSlot函数,利用MyCom.readAll()读取串口的数据,最后显示到文本框内。

    //精简版
    void MyCOM::MyComRevSlot()
    {QByteArray MyComRevBUff;//接收数据缓存QString StrTemp, StrTimeDate, StrTemp1;//读取串口接收到的数据,并格式化数据MyComRevBUff = MyCom.readAll();StrTemp = QString::fromLocal8Bit(MyComRevBUff);curDateTime = QDateTime::currentDateTime();StrTimeDate = curDateTime.toString("[yyyy-MM-dd hh:mm:ss.zzz]");StrTemp = MyComRevBUff.toHex().toUpper();//转换为16进制数,并大写for (int i = 0; i < StrTemp.length(); i += 2)//整理字符串,即添加空格{StrTemp1 += StrTemp.mid(i, 2);StrTemp1 += " ";}//添加时间头StrTemp1.prepend(StrTimeDate);StrTemp1.append("\r\n");//后面添加换行ui.TextRev_2->insertPlainText(StrTemp1);//显示数据ui.TextRev_2->moveCursor(QTextCursor::End);//光标移动到文本末尾
    }
    

image-20241218164558507

六、周期循环发送指令

通过定时器,实现周期性指令发送功能

  • 创建定时器 QTimer* PriecSendTimer;

  • 在构造函数中注册定时器超时connect函数,调用on_pushButtonSend_clicked()

    	connect(PriecSendTimer, &QTimer::timeout, this, [=]() {on_pushButtonSend_clicked(); });
    
  • 通过信号槽机制,绑定选择框状态变化信号处理函数

    image-20241219135218164

  • 编写选择框变化处理函数

    void MyCOM::on_checkBoxPeriodicSend_stateChanged(int arg1)
    {if (arg1 == false){PriecSendTimer->stop();ui.lineEditTime->setEnabled(true);}else{PriecSendTimer->start(ui.lineEditTime->text().toInt());ui.lineEditTime->setEnabled(false);}
    }
    

image-20241219134212145

七、接收流量统计及状态栏设计

通过设计状态栏来实时展示QLabel的相关数据

  • 自定义变量

    	//添加自定义变量long ComSendSum, ComRevSum;//发送和接收流量统计变量QLabel* qlbSendSum, * qlbRevSum;//发送接收流量label对象QLabel* myLink, * MySource;
    
  • 变量绑定状态栏

    //创建底部状态栏及其相关部件
    QStatusBar* STABar = statusBar();qlbSendSum = new QLabel(this);
    qlbRevSum = new QLabel(this);
    myLink = new QLabel(this);
    MySource = new QLabel(this);
    myLink->setMinimumSize(90, 20);// 设置标签最小大小
    MySource->setMinimumSize(90, 20);
    qlbSendSum->setMinimumSize(100, 20);
    qlbRevSum->setMinimumSize(100, 20);
    ComSendSum = 0;
    ComRevSum = 0;setNumOnLabel(qlbSendSum, "Tx: ", ComSendSum);
    setNumOnLabel(qlbRevSum, "Rx: ", ComRevSum);STABar->addPermanentWidget(qlbSendSum);// 从右往左依次添加
    STABar->addPermanentWidget(qlbRevSum);
    STABar->addWidget(myLink);// 从左往右依次添加
    STABar->addWidget(MySource);myLink->setOpenExternalLinks(true);//状态栏显示官网、源码链接
    myLink->setText("<style> a {text-decoration: none} </style> <a href=\"http://8.134.156.7/\">--个人博客--");
    MySource->setOpenExternalLinks(true);
    MySource->setText("<style> a {text-decoration: none} </style> <a href=\"https://github.com/say-Hai/MyCOMDemo\">--源代码--");
    
  • 自定义函数来更改自定义变量

    void MyCOM::setNumOnLabel(QLabel* lbl, QString strS, long num)
    {QString strN = QString("%1").arg(num);QString str = strS + strN;lbl->setText(str);
    }
    
  • 在发送/接收函数中调用自定义函数

    //发送
    temp = MyCom.write(comSendData);
    ComSendSum++;
    setNumOnLabel(qlbSendSum, "Tx: ", ComSendSum);//接收
    MyComRevBUff = MyCom.readAll();
    StrTemp = QString::fromLocal8Bit(MyComRevBUff);
    ComRevSum++;
    setNumOnLabel(qlbRevSum, "Rx: ", ComRevSum);
    

八、数据区清空功能

void MyCOM::on_pushButtonClearRev_clicked()
{ui.TextRev_2->clear();ComSendSum = 0;ComRevSum = 0;setNumOnLabel(qlbSendSum, "Tx: ", ComSendSum);setNumOnLabel(qlbRevSum, "Rx: ", ComRevSum);
}void MyCOM::on_pushButtonClearSend_clicked()
{ui.TextSend_2->clear();ComSendSum = 0;ComRevSum = 0;setNumOnLabel(qlbSendSum, "Tx: ", ComSendSum);setNumOnLabel(qlbRevSum, "Rx: ", ComRevSum);
}

九、文件保存与读取功能

通过文件的读取快速实现对串口发送数据,通过写入文件的方式保存串口的输出。

  • 读取文件:通过QFile aFile(aFileName);QByteArray text = aFile.readAll();来获取文本数据,并写入到文本框中。

    //首先创建on_pushButtonRdFile_clicked信号槽机制打开文件夹选择文件路径
    void MyCOM::on_pushButtonRdFile_clicked()
    {QString curPath = QDir::currentPath();QString dlgTitle = "打开一个文件"; //对话框标题QString filter = "文本文件(*.txt);;所有文件(*.*)"; //文件过滤器QString aFileName = QFileDialog::getOpenFileName(this, dlgTitle, curPath, filter);if (aFileName.isEmpty())return;openTextByIODevice(aFileName);
    }
    //通过openTextByIODevice来读取文件
    bool MyCOM::openTextByIODevice(const QString& aFileName)
    {QFile aFile(aFileName);if (!aFile.exists()) //文件不存在return false;if (!aFile.open(QIODevice::ReadOnly | QIODevice::Text))return false;QByteArray text = aFile.readAll();QString strText = byteArrayToUnicode(text);//编码格式转换,防止GBK中文乱码ui.TextSend_2->setPlainText(strText);aFile.close();return  true;
    }
    //其中防止编码格式问题,通过byteArrayToUnicode进行编码格式转换
    QString MyCOM::byteArrayToUnicode(const QByteArray& array)
    {QTextCodec::ConverterState state;// 先尝试使用utf-8的方式把QByteArray转换成QStringQString text = QTextCodec::codecForName("UTF-8")->toUnicode(array.constData(), array.size(), &state);// 如果转换时无效字符数量大于0,说明编码格式不对if (state.invalidChars > 0){// 再尝试使用GBK的方式进行转换,一般就能转换正确(当然也可能是其它格式,但比较少见了)text = QTextCodec::codecForName("GBK")->toUnicode(array);}return text;
    }
    
  • 写入文件:选择文件路径->调用aFile.write(strBytes, strBytes.length()); 写入文件

    void MyCOM::on_pushButtonSaveRev_clicked()
    {QString curFile = QDir::currentPath();QString dlgTitle = " 另存为一个文件 "; //对话框标题QString filter = " 文本文件(*.txt);;所有文件(*.*);;h文件(*.h);;c++文件(*.cpp) "; //文件过滤器QString aFileName = QFileDialog::getSaveFileName(this, dlgTitle, curFile, filter);if (aFileName.isEmpty())return;saveTextByIODevice(aFileName);
    }
    bool MyCOM::saveTextByIODevice(const QString& aFileName) {QFile aFile(aFileName);if (!aFile.open(QIODevice::WriteOnly | QIODevice::Text))return false;QString str = ui.TextRev_2->toPlainText();//整个内容作为字符串QByteArray  strBytes = str.toUtf8();//转换为字节数组aFile.write(strBytes, strBytes.length());  //写入文件aFile.close();return true;
    }
    

十、多行发送功能

通过信号槽机制和定时器功能,实现对多行数据选择的循环发送

具体逻辑:根据选择框的状态确定定时器状态->通过定时器超时函数唤醒发送事件->在发送事件中确定此次需要发送的行数据->调用对应发送按钮函数

  • 通过选择框的状态变化来打开/关闭定时器发送

    void MyCOM::on_checkBoxMuti_stateChanged(int arg)
    {if (!arg){PriecSendTimer->stop();//关闭定时器ui.lineEditTime->setEnabled(true);//使能对话框编辑}else{LastSend = 0;//从第一行开始发送ui.checkBoxPeriodicSend->setChecked(false);PriecSendTimer->start(ui.lineEditTime->text().toInt());ui.lineEditTime->setEnabled(false);//关闭对话框编辑}
    }
    
  • 重构定时器超时响应函数,适配多行重复发送功能

    connect(PriecSendTimer, &QTimer::timeout, this, [=]() {Pre_on_pushButtonSend_clicked(); });void MyCOM::Pre_on_pushButtonSend_clicked()
    {if (ui.checkBoxPeriodicMutiSend_2->isChecked() == true){while (LastSend < 10){if (checkBoxes[LastSend]->isChecked()){//发送对应行的数据on_pushButtonMuti_clicked(++LastSend);break;}LastSend++;}if (LastSend == 10){LastSend = 0;}}else{//普通发送on_pushButtonSend_clicked();}
    }
    
  • 通过行索引触发对应的点击事件

    void MyCOM::on_pushButtonMuti_clicked(int lineEditIndex)
    {QString Strtemp;switch (lineEditIndex) {case 1:Strtemp = ui.lineEditMuti1_2->text();break;case 2:Strtemp = ui.lineEditMuti2_2->text();break;//...后面对应的操作default:return;  // 默认情况下不做任何操作}ui.TextSend_2->clear();ui.TextSend_2->insertPlainText(Strtemp);ui.TextSend_2->moveCursor(QTextCursor::End);MyCOM::on_pushButtonSend_clicked();
    }
    

    十一:自动刷新串口下拉框

    实现方法:新建一个类继承QComboBox类,重写鼠标点击事件使其调用扫描端口函数

  • 新建mycombobox类,继承QComBox

    #include <QComboBox>
    #include <QMouseEvent>
    #include <QSerialPort>
    #include <QSerialPortInfo>class mycombobox : public QComboBox
    {Q_OBJECT
    public:explicit mycombobox(QWidget* parent = nullptr);void mousePressEvent(QMouseEvent* event) override;
    signals:
    private:void scanActivatePort();
    };
  • 重写扫描函数和鼠标点击函数

    mycombobox::mycombobox(QWidget* parent) : QComboBox(parent)
    {scanActivatePort();
    }void mycombobox::mousePressEvent(QMouseEvent* event)
    {if (event->button() == Qt::LeftButton){scanActivatePort();showPopup();}
    }void mycombobox::scanActivatePort()
    {clear();//创建串口列表QStringList comPort;foreach(const QSerialPortInfo & info, QSerialPortInfo::availablePorts()){QString serialPortInfo = info.portName() + ": " + info.description();// 串口设备信息,芯片/驱动名称comPort << serialPortInfo;}this->addItems(comPort);
    }
    
  • 最后将comboBoxNo_2组件提升为mycombobox

image-20241220115619125

到此整个软件设计完毕

END:信号槽绑定图

image-20241220120509952

参考文献:

[1] https://rymcu.com/portfolio/40


http://www.ppmy.cn/server/154056.html

相关文章

区块链平台安全属性解释

区块链平台安全属性解释 双向认证 解释:双向认证是指在通信过程中,**通信双方都需要对对方的身份进行验证,确保对方是合法的、可信任的实体。**只有双方身份都得到确认后,通信才会被允许进行,从而防止非法用户的接入和数据的窃取或篡改。举例:在基于区块链和联邦学习的数…

【ELK】filebeat采集数据输出到kafka指定topic

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 背景filebeat主体配置filebeat.inputs部分filebeat.output部分 filebeat完整配置 背景 今天收到需求&#xff0c;生产环境中通需要优化filebeat的输出&#xff0c;…

StarRocks 存算分离在得物的降本增效实践

编者荐语&#xff1a; 得物优化数据引擎布局&#xff0c;近期将 4000 核 ClickHouse 迁移至自建 StarRocks&#xff0c;成本降低 40%&#xff0c;查询耗时减半&#xff0c;集群稳定性显著提升。本文详解迁移实践与成果&#xff0c;文末附丁凯剑老师 StarRocks Summit Asia 2024…

设计一个监控摄像头物联网IOT(webRTC、音视频、文件存储)

前言&#xff1a; 设计一个完整的 监控摄像头物联网 IoT 平台 涉及 视频直播和点播、WebRTC 和 文件存储模块&#xff0c;可以分为以下几个主要部分&#xff1a;摄像头设备、服务端处理、Web 前端、视频流存储和回放。以下是结合这些技术的一个具体完整流程设计&#xff0c;涵盖…

python笔记随记

前言 笔记记录 内容 1.关键词条件&#xff1a; 代码块 2.总是只有一个if语句被执行&#xff0c;所有需要的elif语句在if语句之后&#xff0c;它们的先后顺序很重要&#xff0c;当前边的的if或elif已经执行则下边的elif则全部跳过&#xff0c;如果希望确保一个语句被执行则…

OpenAI 12天发布会:AI革命的里程碑@附35页PDF文件下载

在人工智能的浪潮中&#xff0c;OpenAI的12天发布会无疑是2024年科技界的一场盛宴。从12月5日开始&#xff0c;OpenAI连续12天每天发布一个新应用或功能&#xff0c;标志着AI技术的又一次飞跃。本文将梳理这些激动人心的发布&#xff0c;带你一探究竟。 OpenAI发布会概览 Ope…

Qt6 QML RegularExpressionValidator 输入中文的坑

本人最近使用Qt6.7.2编译CMAKE项目&#xff0c;主风格qml &#xff0c;但在用TextField{}时却遇到了问题&#xff0c; 原本要求&#xff0c;编辑框只能输入中文汉字&#xff0c; 一听&#xff0c;很简单嘛&#xff0c;正则表达式&#xff1a; 一看&#xff0c;没毛病&#xf…

Python 爬虫中的反爬策略及详细应对方法

在构建Python爬虫的过程中&#xff0c;网站为了保护自身资源和用户体验&#xff0c;常常会采取一系列反爬策略来限制或阻止自动化程序的访问。了解这些策略对于设计更智能、更合规的爬虫至关重要。以下是详细的反爬措施及其应对方法&#xff1a; 1. User-Agent 检测 策略描述…