QT-串口调试助手自动定时收发、十六进制转换

2023-05-16

这篇调试助手比较详细:不仅有十六进制转换、串口自动识别还有自动发送等功能。
程序链接--欢迎关注哦。。https://download.csdn.net/download/m0_46436890/13793486)
一、安装QTCreator
官网自行安装即可,我安装的是QT5.12.8,目测还不错,网上评价QT5.12.9很好用。可以尝试下载。

二、安装编译器
如果如果没有特殊编译器要求,可以直接使用自带的MinGW的64位编译器,也可以安装Visual Studio配置使用它的编译器,可以编译64位的。编译器会自动检测的,建议先安装Visual Studio,QT安装时会自动识别到。
在这里插入图片描述
三、创建项目工程
此处省略(不浪费大家时间)........不懂可以自行百度
废话不多说直接上图..调试界面
在这里插入图片描述
四、程序编写
1、函数入口main.c
mian.cpp中则实例化了Dialog,并调用了show函数
程序通过main函数入口开始执行,于是UI界面就显示出来了

#include "dialog.h"

#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Dialog w;
    w.show();
    return a.exec();
}

2、主要函数以及调用库函数
(1).在项目.pro文件中加入serialport

QT       += core gui
QT       += serialport

(2).引入qt中串口通信和控件等需要的头文件

#include "dialog.h"
#include "ui_dialog.h"
#include <QtSerialPort/QtSerialPort>
#include <QSerialPortInfo>
#include <QMessageBox>
#include<QString>
#include <QDebug>
#include<QIcon>
#include<QSettings>
#include <QCheckBox>
#include <QGroupBox>

3、配置串口初始化
(1).设置串口基本信息。波特率、数据位、奇偶校验等。

static const char blankString[] = QT_TRANSLATE_NOOP("SettingsDialog", "N/A");
Dialog::Dialog(QWidget *parent)
    : QDialog(parent)
    , ui(new Ui::Dialog)
{
     ui->setupUi(this);
     serial = new QSerialPort;
     //ui->portNameBox->addItem(tr("custom"));
     //设置波特率
     ui->baudrateBox->addItem(QStringLiteral("9600"), QSerialPort::Baud9600);
     ui->baudrateBox->addItem(QStringLiteral("19200"), QSerialPort::Baud19200);
     ui->baudrateBox->addItem(QStringLiteral("38400"), QSerialPort::Baud38400);
     ui->baudrateBox->addItem(QStringLiteral("115200"), QSerialPort::Baud115200);
     ui->baudrateBox->addItem(tr("Custom"));
    //设置数据位
     ui->dataBitsBox->addItem(QStringLiteral("5"), QSerialPort::Data5);
     ui->dataBitsBox->addItem(QStringLiteral("6"), QSerialPort::Data6);
     ui->dataBitsBox->addItem(QStringLiteral("7"), QSerialPort::Data7);
     ui->dataBitsBox->addItem(QStringLiteral("8"), QSerialPort::Data8);
     ui->dataBitsBox->setCurrentIndex(3);
     //设置奇偶校验位
     ui->ParityBox->addItem(tr("None"), QSerialPort::NoParity);
     ui->ParityBox->addItem(tr("Even"), QSerialPort::EvenParity);
     ui->ParityBox->addItem(tr("Odd"), QSerialPort::OddParity);
     ui->ParityBox->addItem(tr("Mark"), QSerialPort::MarkParity);
     ui->ParityBox->addItem(tr("Space"), QSerialPort::SpaceParity);
     //设置停止位
     ui->stopBitsBox->addItem(QStringLiteral("1"), QSerialPort::OneStop);
     ui->stopBitsBox->addItem(QStringLiteral("2"), QSerialPort::TwoStop);
     //添加流控
     ui->stopBitsBox->addItem(tr("None"), QSerialPort::NoFlowControl);
     ui->stopBitsBox->addItem(tr("RTS/CTS"), QSerialPort::HardwareControl);
     ui->stopBitsBox->addItem(tr("XON/XOFF"), QSerialPort::SoftwareControl);
     //禁用发送按钮
     ui->sendButton->setEnabled(false);
     ui->timeSend->setEnabled(false);
}

Dialog::~Dialog()
{
    delete ui;
}

(2).刷新串口COM,获得最新串口号

void Dialog::on_getSerial_Btn_clicked()
{
    QString description;
    QString manufacturer;
    QString serialNumber;
    //获取可以用的串口
    QList<QSerialPortInfo> serialPortInfos = QSerialPortInfo::availablePorts();
    //输出当前系统可以使用的串口个数
   qDebug() << "Total numbers of ports: " << serialPortInfos.count();
   ui->portNameBox->clear();
   //将所有可以使用的串口设备添加到ComboBox中
    for (const QSerialPortInfo &serialPortInfo : serialPortInfos)
    {
     //ui->portNameBox->clear();
     QStringList list;
     description = serialPortInfo.description();
     manufacturer = serialPortInfo.manufacturer();
     serialNumber = serialPortInfo.serialNumber();
     list << serialPortInfo.portName()
       << (!description.isEmpty() ? description : blankString)
       << (!manufacturer.isEmpty() ? manufacturer : blankString)
       << (!serialNumber.isEmpty() ? serialNumber : blankString)
       << serialPortInfo.systemLocation()
       << (serialPortInfo.vendorIdentifier() ? QString::number(serialPortInfo.vendorIdentifier(), 16) : blankString)
       << (serialPortInfo.productIdentifier() ? QString::number(serialPortInfo.productIdentifier(), 16) : blankString);
     ui->portNameBox->addItem(list.first(), list);
    }
}

(3).打开串口
  加入一个打开关闭串口的按钮,文本显示“打开串口”时,点击可以关闭串口。文本显示“关闭串口“则相反。


void Dialog::on_openButton_clicked()
{
 qDebug() << ui->openButton->text();
 if (ui->openButton->text() == tr("打开串口"))
 {

   //设置串口名字
   serial->setPortName(ui->portNameBox->currentText());
   //设置波特率
   serial->setBaudRate(ui->baudrateBox->currentText().toInt());
   //设置数据位
   serial->setDataBits(QSerialPort::Data8);
   //设置奇偶校验位
   serial->setParity(QSerialPort::NoParity);
   //设置停止位
   serial->setStopBits(QSerialPort::OneStop);
   //设置流控
   serial->setFlowControl(QSerialPort::NoFlowControl);
   //打开串口
   if (serial->open(QIODevice::ReadWrite))
   {
    //QMessageBox::information(this, "提示", "打开成功!");
    ui->baudrateBox->setEnabled(false);
    ui->dataBitsBox->setEnabled(false);
    //ui->comboBox_flowBit->setEnabled(false);
    ui->ParityBox->setEnabled(false);
    ui->portNameBox->setEnabled(false);
    ui->stopBitsBox->setEnabled(false);
    ui->sendButton->setEnabled(true);
    ui->timeSend->setEnabled(true);
    ui->openButton->setText(tr("关闭串口"));
    //信号与槽函数关联
    connect(serial, &QSerialPort::readyRead, this, &Dialog::readData);
   }
   else
   {
    QMessageBox::about(this,tr("提示信息"),tr("请打开串口"));

   }
 }
 else
  {
   //关闭串口
   //serial->clear();
   serial->close();
   //mIsOpen = true;
   //ui->openButton->setText("关闭");
   //qDebug() << "成功打开串口" << mPortName;
   //serial->deleteLater();
   //恢复设置功能
   ui->baudrateBox->setEnabled(true);
   ui->dataBitsBox->setEnabled(true);
  // ui->comboBox_flowBit->setEnabled(true);
   ui->ParityBox->setEnabled(true);
   ui->portNameBox->setEnabled(true);
   ui->stopBitsBox->setEnabled(true);
   ui->openButton->setText(tr("打开串口"));
   ui->sendButton->setEnabled(false);
  }
}

4、读取串口数据
  读取数据内容,然后进行数据处理,使输出数据即可以是中文也可以输出十六进制
而且在UI里面添加了QCheckBox控件。点击控件来判断是否输出十六进制


void Dialog::readData()
{
   QByteArray buf;
   QString res;

   buf=serial->readAll();
   res=QString(buf);

   if(!buf.isEmpty())
   {
       // 调试输出buf大小
               qDebug()<<buf.size()<<endl;

               // 将QByteArray数据类型转换,要能正确显示中文,需要使用QString::fromLocal8Bit
               //        QString str = QString::fromUtf8( buf );
               QString str = QString::fromLocal8Bit( buf );

               qDebug()<< str <<endl;

               // 如果以16进制显示数据:
               if (ui->HexResv_chk->isChecked())
               {
                   QString hex_data = buf.toHex().data(); // 将buf里的数据转换为16进制
                   hex_data = hex_data.toUpper(); // 转换为大写
                   QString hex_str;
                   // 将16进制转换为大写
                   for (int i=0; i< hex_data.length(); i+=2)
                   {
                       QString st = hex_data.mid(i,2);
                       hex_str+=st;
                       hex_str+=' ';
                   }
                   ui->resvEdit->append(hex_str);
               }
               else
               {
                   ui->resvEdit->append(str);
               }
   }
buf.clear();
}

5、发送数据以及定时发送数据
简单的发送数据没有什么要额外配置的,调用write函数就可以,但是如果输出十六进制数需要添加sendButton的QCheckBox控件。
发送数据处理十六进制需要调用两个函数

    void StringToHex(QString str, QByteArray &senddata);
    char ConvertHexChar(char ch);

(1).普通发送数据

void Dialog::on_sendButton_clicked()  //发送数据
{
    QByteArray buf;
    QString sendData;

    buf=ui->sendEdit->toPlainText().toLatin1();
    sendData=QString(buf);
    if(!buf.isEmpty())
    {
        QString str = QString::fromLocal8Bit( buf );

        qDebug()<< str <<endl;
        if (ui->HexSend_Chk->isChecked())
        {
            QString str = ui->sendEdit->toPlainText();//从LineEdit得到字符串
            QByteArray senddata;
            StringToHex(str,senddata);//将str字符串转换为16进制的形式
            serial->write(senddata);
        }
        else
        {
            serial->write(str.toLocal8Bit());// 要能正确发送中文字符,需要使用QString的toLocal8Bit方法
       //    serial->write(str.toLatin1());
        }
    }
buf.clear();
}

(2).定时发送数据
  使用定时发送功能需要在Dialog.h里面添加#include 头文件

//自动发送-按时()ms
void Dialog::on_timeSend_clicked()
{
      ui->sendButton->setEnabled(false);
      int time;
      static bool en=true; //标识. 是否设置自动发送时间
        if(ui->lineEdit->text()==NULL)
           QMessageBox::about(this,tr("提示信息"),tr("请设置时间"));
        else
        {
            if(en==true)
            {
            //QMessageBox::about(this,tr("提示信息"),tr("已开始自动发送"));
            time=ui->lineEdit->text().toInt();  //获取自动发送的时间间隔
            ui->lineEdit->setEnabled(false);
            timeSend =new QTimer();
            timeSend->setInterval(1000);
            connect(timeSend,&QTimer::timeout,this,[=](){on_sendButton_clicked();
            });

            timeSend->start(time);
            en=false;
            }
            else if(en==false)
            {
                timeSend->stop();
                //QMessageBox::about(this,tr("提示信息"),tr("已停止自动发送"));
                en=true;
                ui->lineEdit->setEnabled(true);
                ui->sendButton->setEnabled(true);
            }
         }     
}

(3).十六进制处理函数

void Dialog::StringToHex(QString str, QByteArray &senddata)
{
    int hexdata,lowhexdata;
    int hexdatalen = 0;
    int len = str.length();
    senddata.resize(len/2);
    char lstr,hstr;
    for(int i=0; i<len; )
    {
        //char lstr,
        hstr=str[i].toLatin1();
        if(hstr == ' ')
        {
            i++;
            continue;
        }
        i++;
        if(i >= len)
            break;
        lstr = str[i].toLatin1();
        hexdata = ConvertHexChar(hstr);
        lowhexdata = ConvertHexChar(lstr);
        if((hexdata == 16) || (lowhexdata == 16))
            break;
        else
            hexdata = hexdata*16+lowhexdata;
        i++;
        senddata[hexdatalen] = (char)hexdata;
        hexdatalen++;
    }
    senddata.resize(hexdatalen);
}

char Dialog::ConvertHexChar(char ch)
{
    if((ch >= '0') && (ch <= '9'))
            return ch-0x30;
        else if((ch >= 'A') && (ch <= 'F'))
            return ch-'A'+10;
        else if((ch >= 'a') && (ch <= 'f'))
            return ch-'a'+10;
        else return (-1);
}

五、最后附上UI设计控件的命名和布局。
在这里插入图片描述
全是干货............有问题欢迎留言。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

QT-串口调试助手自动定时收发、十六进制转换 的相关文章

随机推荐

  • 你是我的眼:水哥王昱珩带你重新打量这世界

    水哥 王昱珩在 最强大脑 舞台上凭借着 微观辨水 一战成名 在节目中 xff0c 他以1秒四杯的速度从520杯同质同量同水源的水中 xff0c 迅速找出了之前被随机选中的那杯 xff0c 甚至看出了这杯水从被选中拿起观察再到放回原位 xff
  • 性能优化与内存优化

    性能优化 主要以这四个方向进行优化 xff1a 稳定流畅耗损apk 瘦身 稳定 避免内存溢出异常捕获反馈机制 流畅 卡顿的原因 xff1a 布局复杂动画过多View 的过度绘制UI 耗时操作频繁 GC 耗损 减少没必要的网络访问或合并相关网
  • (2)工作空间

    工作空间 特点创建工作空间编译添加环境变量 特点 1 工作空间的主要作用是存放工程文件和文件代码 xff08 存放工作空间和代码的一个文件夹 xff09 2 ROS的开发依赖于工作空间 3 编写源代码 编译都是在工作空间下完成的 4 一般工
  • 工程伦理 清华大学 MOOC 期末答案 2020冬

    前言 xff1a 答案是本人自己查阅 不保证其完全正确性 xff0c 仅供参考 点个赞再走呗 xff1a xff09 1 单选题 1分 以下不属于安全文化的核心的是 D A 人的安全知识 B 人的安全意识 C 人的安全技能 D 人的安全装备
  • HTML移动端的基本了解

    文章目录 前言 一 PC端与移动端的区别 PC端 xff1a 移动端 xff1a 二 视口 布局视口 视觉视口 理想视口 meta视口标签 三 二倍图 四 移动端开发选择 单独制作移动端页面 主流 响应式页面兼容移动端 其次 总结 前言 分
  • 树莓派4B(armv7l,arm32)buster部署英特尔第二代神经计算棒,示例:运行darknet-yolov4-tiny

    记录整个部署的步骤 英特尔第二代神经计算棒 xff08 Intel NCS2 xff09 的使用需要到OpenVINO官网下载安装某一个版本的软件 xff0c 目前是支持Windows Linux和树莓派的Raspbian OS 下面贴出来
  • 树莓派结合英特尔神经计算棒二代(NCS2)的openvino包部署人工智能应用

    利用树莓派和英特尔神经计算棒二代 xff08 NCS2 xff09 进行边缘端AI硬件平台部署三部曲 xff1a 二 xff09 树莓派4B和NCS2配置时cmake编译报错以及import error can t find moudle
  • 树莓派报错“Cannot currently show the desktop”的完美解决办法

    最近在利用树莓派部署神经网络的时候出现了一些大大小小的问题 xff0c 很多问题都可以在网上直接或间接地找到答案 xff0c 但有个别问题即使按照网上的高赞博客说的去做了仍然没用 笔者根据最近遇到的有关树莓派VNC win10远程桌面连接
  • Nvidia Jetson XAvier NX开发套件从装机到pytorch环境搭建YoloV5+DeepSort+TensorRT

    目录 1 刷机与装机1 1 准备VMware工作站和linux的unbuntu16 04虚拟机 xff1a 1 2 将SD上的系统移动至SSD1 3 SSH配置1 4 查看Jetpack版本1 5 启动风扇 2 深度学习环境配置2 1 py
  • Python+OpenCV+matplotlib+wxPython实现的图像处理程序

    Python 43 OpenCV 43 matplotlib 43 wxPython实现的图像处理程序 背景 xff1a 本学期多媒体技术原理与应用课程的实验部分代码结果截图结语 背景 xff1a 本学期多媒体技术原理与应用课程的实验 本学
  • 按量购买阿里云服务器、k8s集群

    基础配置 创建私有网络 注意这里的交换区必须和服务器的所选区相同 连接测试 ping a 172 31 0 141 172 31 0 143 172 31 0 142 ping一下私有网络 ping 172 32 0 141 设置弹性ip
  • 小米6渲染图曝光 边框惊艳 国产曲面旗舰要发黑色版

    昨天咱们刚刚说过 小米6会有陶瓷版本 并且估计会在四月发布 今天 网友的渲染图又来了 没上车的赶紧上车 别忘了底部打卡 最近 对于小米6的传言很众多 xff0c 但不管怎么说 xff0c 骁龙835处理器 IMX400传感器以及高屏占比等都
  • 时间戳对齐的简单算法(原创)

    实际融合算法过程中常常出现时间戳对不齐的情况 为了解决此问题 xff0c 用了两个队列来解决时间戳匹配问题 时间戳对齐算法实现原理如下下图为插入状态 如图所示共有两个队列 xff0c 有七个状态 xff0c 通过对于轮速递推结果的动态保存
  • java中equals和==的区别(简单介绍)

    java中equals和 61 61 的区别 简单介绍 equals方法是java lang Object类的方法 有两种用法说明 一 对于字符串变量来说 xff0c 使用 61 61 和 equals 方法比较字符串时 xff0c 其比较
  • 如何上传到GitHub的main分支而不是master分支

    由于在2020年10月01日后 xff0c 在 GitHub 上创建的仓库都默认命名为 main 而非原本的 master 所以 有人看了我的 https blog csdn net m0 46419510 article details
  • Python中的Pandas、Matplotlib库详解

    文章目录 Pandas数据分析库Pandas的基本数据结构访问数据算术运算和对齐数据整理 Matplotlib绘图库Matplotlib简介几种常见的图形多图绘制使用Pandas绘图 Pandas数据分析库 1 xff09 Pandas是目
  • 卡尔曼滤波模型及Matlab模型建立

    目录 一 卡尔曼滤波 1 概念解析 xff1a 2 卡尔曼滤波的最优估计模型 3 实例 小车匀加速直线运动 4 Matlab建模 二 扩展卡尔曼滤波 xff08 EKF Extended KAlman Filter xff09 1 非线性系
  • 智能车 PID 调试

    智能车 PID 调试 文章目录 智能车 PID 调试学习目的开环控制与闭环控制开环控制闭环控制小结 PID 概述简介PID 公式 xff1a 舵机 PID分析算法 电机 PID分析算法调试口决 注意事项 学习目的 使电机速度和舵机转向更精准
  • ORBSLAM2系统学习(二)

    提示 xff1a 文章写完后 xff0c 目录可以自动生成 xff0c 如何生成可参考右边的帮助文档 目录 前言 一 ORBSLAM2简介 二 系统综述 系统框架 追踪线程Tracking 局部建图线程local mapping 回环检测l
  • QT-串口调试助手自动定时收发、十六进制转换

    这篇调试助手比较详细 xff1a 不仅有十六进制转换 串口自动识别还有自动发送等功能 程序链接 xff0d xff0d 欢迎关注哦 https download csdn net download m0 46436890 13793486