发新帖本帖赏金 10.00元(功能说明)我要提问
返回列表
打印
[程序源码]

开源的Qt 串口助手 一学就会

[复制链接]
3668|8
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
本帖最后由 一路向北lm 于 2021-1-22 10:50 编辑

#申请原创#         

      串口调试助手是一款用于串口调试的工具,目前网上存在很多个版本,功能都差不多,但稳定性有好有坏,如果只用于一般的串口调试,这些工具够用了,如果想开发一款适于自己的串口调试助手,本文也许可以帮你。本文中的调试助手用QT开发,QT的开发环境不在这里详述了,不会安装的可以在网上找找类似的博文吧,以下仅供参考。


第一部分:代码托管:
此部分可能存在一些未知的bug,欢迎广大网友指出,为了方便管理和学习,所有代码均托管到gitee,链接地址:https://gitee.com/lumengcode/my-qt

第二部分:效果展示:

实现的功能:

1.自动获取计算机的端口号;
2.串口参数可更改:包括 波特率、数据位、停止位、校验和等............
3.串口数据的发送和接收
4.支持十六进制数据的发送和接收
5.支持时间戳功能,方便文件的存储查看
6.发送从窗口和接收窗口的清理
7.定时发送功能
............

简单设置一下背景色,好看多了!

第二部分:代码部分:


1.当我们的计算机的端口号发生改变时,串口助手要具备实时扫面本机的端口号的功能,具有实时获取有效的串口信息,并将其刷新到下拉框中供我们选择。
有些自己编写的串口助手是没有这个功能的,这里我给大家补充上去。
//使用foreach获取有效的串口信息
     foreach(const QSerialPortInfo &info, QSerialPortInfo::availablePorts())
     {
         //这里相当于自动识别串口号之后添加到了cmb,如果要手动选择可以用下面列表的方式添加进去
         Serial.setPort(info);
         if(Serial.open(QIODevice::ReadWrite))
         {
             //将串口号添加到cmb
             ui->comboBox_Port->addItem(info.portName());
             //关闭串口等待人为(打开串口按钮)打开
             Serial.close();
         }
     }
2.填充下拉框的波特率、数据位、停止位、效验位….,初始化下拉框默认参数,这个参数设置大部分的串口助手都会具备,因此不足为奇。
该有的功能个咱还是得有的。
// 填充波特率
    QStringList Baud;
    Baud<<"1200"<<"2400"<<"4800"<<"9600"<<"38400"<<"115200";
    ui->comboBox_Baud->addItems(Baud);
    // 填充数据位
    QStringList DataBit;
    DataBit<<"5"<<"6"<<"7"<<"8";
    ui->comboBox_DataBit->addItems(DataBit);
    // 填充停止位
    QStringList StopBit;
    StopBit<<"1"<<"1.5"<<"2";
    ui->comboBox_StopBit->addItems(StopBit);
    // 填充效验位
    QStringList CheckBit;
    CheckBit<<"奇效验"<<"偶效验"<<"无";
    ui->comboBox_CheckBit->addItems(CheckBit);

    //初始化默认参数
    ui->comboBox_Baud->setCurrentIndex(3); //默认9600
    ui->comboBox_DataBit->setCurrentIndex(3); //默认8bit Data
    ui->comboBox_StopBit->setCurrentIndex(0); //默认1bit Stop
    ui->comboBox_CheckBit->setCurrentIndex(2); //默认 无效验
3.串口打开和关闭按钮操作,这个就是打开串口按钮和关闭按钮的逻辑操作,成功打开串口后,相应的参数将会被设置。串口即可以用于
数据的发送和接收了,这里也处理,打开失败时的逻辑操作,可谓是“疏而不漏也!”。
//串口打开和关闭按钮
void MainWindow::on_pushButton_Open_clicked()
{
    //设置串口号;也就是说打开的是当前显示的串口
    if(ui->comboBox_Port->currentText().isEmpty())
    {
        QMessageBox::information(this,"提示","没有可用的串口");
        return;
    }
    Serial.setPortName(ui->comboBox_Port->currentText());
    if(ui->pushButton_Open->text() == "打开串口")
    {
           if(Serial.open(QIODevice::ReadWrite))//读写方式打开,成功后设置串口
           {
               //设置波特率
               Serial.setBaudRate(ui->comboBox_Baud->currentText().toInt());

               //设置数据位
               switch(ui->comboBox_DataBit->currentText().toInt())
               {
                     case 5:
                             Serial.setDataBits(QSerialPort::Data5);
                     break;
                     case 6:
                             Serial.setDataBits(QSerialPort::Data6);
                     break;
                     case 7:
                              Serial.setDataBits(QSerialPort::Data7);
                     break;
                     case 8:
                              Serial.setDataBits(QSerialPort::Data8);
                     break;
                     default:
                              QMessageBox::information(this,"提示","数据位配置出错");
                              return;
                     break;
               }

               //设置校验位
               if (ui->comboBox_CheckBit->currentText() == "奇效验")
               {
                   Serial.setParity(QSerialPort::OddParity);
               }
               else if (ui->comboBox_CheckBit->currentText() == "偶效验")
               {
                   Serial.setParity(QSerialPort::EvenParity);
               }
               else if (ui->comboBox_CheckBit->currentText() == "无")
               {
                   Serial.setParity(QSerialPort::NoParity);
               }

               //设置停止位
               if (ui->comboBox_StopBit->currentText().toFloat() == 1)
               {
                   Serial.setStopBits(QSerialPort::OneStop);
               }
               else if(ui->comboBox_StopBit->currentText().toFloat() == 1.5)
               {
                   Serial.setStopBits(QSerialPort::OneAndHalfStop);
               }
               else if(ui->comboBox_StopBit->currentText().toFloat() == 2)
               {
                   Serial.setStopBits(QSerialPort::TwoStop);
               }

               //设置流控制
               Serial.setFlowControl(QSerialPort::NoFlowControl);
               ui->pushButton_Open->setText("关闭串口");


               //建立串口接收的槽函数
               connect(&Serial,&QSerialPort::readyRead ,this,&MainWindow::ReadRecData);

              // timer0->start(100);

            }
           else//串口打开失败
           {
               QMessageBox::about(NULL, "提示", "打开出错,串口被占用!");
               return ;
           }
    }
    else if(ui->pushButton_Open->text() == "关闭串口")
    {
        Serial.close();//关串口
        //timer0->stop();
        ui->pushButton_Open->setText("打开串口");
    }
}
4. 串口接收数据函数(支持时间戳、HEX接收) 这个是很关键的地方了,要保证数据接收的完整性和实时性,可采用两种
接收数据的模式:定时器触发和槽触发,定时器触发我这里采用的是100ms的中断接收,大家还可以调的更小一点。
void MainWindow::ReadRecData()
{
    QByteArray readData = Serial.readAll();//读取串口数据
    QByteArray NewData;
    QString current_date;

   if(readData != NULL)//将读到的数据显示到数据接收区
   {
       if(HexRecvFlag)  //判断是否使用HEX
       {
           //判断是否使用时间戳
           if(EnableTimeFlag == 1)
           {
               current_date_time = QDateTime::currentDateTime();
               current_date += "[";
               current_date += current_date_time.toString("yyyy-MM-dd hh:mm:ss");
               current_date += "]收->";
               ui->textEdit_Recv->append(current_date.toUtf8() + readData.toHex());
           }
           else
           {
              ui->textEdit_Recv->append(readData.toHex());
           }

       }
       else
       {
           //判断是否使用时间戳
           if(EnableTimeFlag == 1)
           {
               current_date_time = QDateTime::currentDateTime();
               current_date += "[";
               current_date += current_date_time.toString("yyyy-MM-dd hh:mm:ss");
               current_date += "]收->";
               ui->textEdit_Recv->append(current_date.toUtf8() + readData);
           }
           else
           {
              ui->textEdit_Recv->append(readData);
           }
       }

   }
}
5. 串口发送数据函数(支持时间戳、HEX接收) 这个是很关键的地方了,串口发送数据的方式就比较简单了,直接将数据送入缓冲区
//发送数据
void MainWindow::on_pushButton_Send_clicked()
{
    QString DataStr;
    QString NewData;
    QString current_date;

    DataStr = ui->textEdit_Send->toPlainText();
    if(ui->pushButton_Open->text() == "打开串口")
    {
       QMessageBox::information(this,"提示","未打开串口");
       return;
    }

    if(EnableTimeFlag == 1)
    {
        current_date_time = QDateTime::currentDateTime();
        current_date += "[";
        current_date += current_date_time.toString("yyyy-MM-dd hh:mm:ss");
        current_date += "]发->";
        NewData = current_date + DataStr;
    }
    else
    {
      NewData = DataStr;
    }

    if(HexSendFlag)
    {
       Serial.write(DataStr.toUtf8().toHex());//写入缓冲区
    }
    else
    {

      ui->textEdit_Recv->append(NewData.toUtf8());
    }
}
6.清除接收和发送窗口数据函数,为了方便调试和观察,这里添加了清除接收和发送窗口数据函数的操作。
//清除接收窗口数据
void MainWindow::on_pushButton_ClearRecv_clicked()
{
   ui->textEdit_Recv->clear();
}

//清除发送窗口数据
void MainWindow::on_pushButton_2_clicked()
{
   ui->textEdit_Send->clear();
}

7.使能时间戳,时间戳的主要目的在于通过一定的技术手段,对数据产生的时间进行认证,从而验证这段数据在产生后是否经过篡改。所以时间戳服务的提供者必须证明服务中使用的时间源是可信的,所提供的时间戳服务是安全的。
void MainWindow::on_checkBox_EnableTime_clicked(bool checked)
{
     if(checked == true)
     {
         EnableTimeFlag = 1;
     }
     else
     {
         EnableTimeFlag = 0;
     }
}

8. 使能定时发送,定时发送很香了,必须得有啊!
void MainWindow::on_checkBox_clicked(bool checked)
{
    if(checked == true)
    {
        if(ui->pushButton_Open->text() == "打开串口")
        {
           QMessageBox::information(this,"提示","未打开串口");
           ui->checkBox->setChecked(false);
           return;
        }
        quint32 stime= ui->lineEdit_STime->text().toInt();
        timer_id1 = startTimer(stime);
        ui->lineEdit_STime->setEnabled(false);
    }
    else
    {
       killTimer(timer_id1);
       ui->lineEdit_STime->setEnabled(true);
    }
}
9. 使能HEX 发送和接收按钮,HEX那是标配,我只希望不要出bug,慢慢完善吧!
void MainWindow::on_checkBox_HexRecv_clicked(bool checked)
{
    if(checked)
    {
        HexRecvFlag = 1;
    }

    else
        HexRecvFlag = 0;
}

void MainWindow::on_checkBox_HexSend_clicked(bool checked)
{
    if(checked)
        HexSendFlag = 1;
    else
        HexSendFlag = 0;
}
10. 定时器中断函数 触发接收串口数据的核心,没它啥也干不了。
void MainWindow:: timerEvent(QTimerEvent *ev)
{
    if(ev->timerId() == timer_id1)
    {
       on_pushButton_Send_clicked();
    }
}
串口助手部分最后的展示效果


@21小跑堂   @21小跑堂    @21小跑堂   








使用特权

评论回复

打赏榜单

21小跑堂 打赏了 10.00 元 2021-01-22
理由:恭喜通过原创文章审核!请多多加油哦!

相关帖子

沙发
海洋无限| | 2021-1-26 12:25 | 只看该作者
可以

使用特权

评论回复
板凳
nvjwiciw659| | 2021-1-29 15:34 | 只看该作者
给师兄顶

使用特权

评论回复
地板
一路向北lm|  楼主 | 2021-1-30 18:22 | 只看该作者

使用特权

评论回复
5
sch_l| | 2021-2-1 08:56 | 只看该作者
高手!

使用特权

评论回复
6
catvevs| | 2021-4-1 09:43 | 只看该作者
谢谢分享

使用特权

评论回复
7
qqq_147258| | 2021-9-10 10:33 | 只看该作者
貌似缺少“res/res.qrc”文件。

使用特权

评论回复
8
qin552011373| | 2021-10-15 10:55 | 只看该作者
收藏备用

使用特权

评论回复
9
qjp1988113| | 2021-10-15 15:27 | 只看该作者
赞一个,楼主有没有后续完善和美化?

使用特权

评论回复
发新帖 本帖赏金 10.00元(功能说明)我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

277

主题

3815

帖子

76

粉丝