串口通信通过Tcp传到服务器实现串口数据转发

2023-11-02

主要目的:通过串口获取外设的数据,转发到服务器进行处理并在Web端显示设备的信息。

主要用到的知识点:

第一:串口通信相关的知识。

1.串口通信用到的两个头文件:

#include <QtSerialPort/QSerialPort>
#include <QtSerialPort/QSerialPortInfo>

QSerialPort:用于访问串口,并对串口进行操作。

QSerialPortInfo:提供了系统中存在的串口的信息。

2.工程文件(.pro)中加下面一行代码:

QT       += serialport

注意:工程文件中添加上述代码后,要保存后才能生效,方法:在QT Creator中文件—保存所有文件。

第二:Tcp通信相关知识。

1.Tcp通信用到的头文件:

#include <QTcpSocket>

2.需要.pro文件中加入下面代码:

QT       += network

 代码实现:

第一:设计模块截图

          下面的红色是各个控件的QbjectNname,在代码要用到,为了方便读者方便阅读所以都标记出来了。下面“发送”按钮是预留按钮,在代码中没有用到,准备在后面的更新中再优化用到此按钮哦。

第二:代码展示

#-------------------------------------------------
#
# Project created by QtCreator 2019-09-30T15:51:56
#
#-------------------------------------------------

QT       += core gui network serialport

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = qianxunSerialPort
TEMPLATE = app


SOURCES += main.cpp\
        widget.cpp

HEADERS  += widget.h

FORMS    += widget.ui

注意:QT += core gui network serialport,一定要添加后在文件中保存,不然会出问题的哦。

widget.h:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpSocket>
#include<QHostAddress>
#include<QDebug>
#include<QSerialPort>
#include<QSerialPortInfo>
#include<QTextCodec>
#include<QByteArray>
namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
    void InitTcp();
    void InitPort();
    void doProcessWrite();
    void doProcessTcpWrite();
    void writeByteArray();
private slots:
    void doProssConnected();
    void doProcessReadyRead();
    void doProssDisconnected();
    void doProcessSerialRead();

private slots:
    void on_pushButton_clicked();

    void on_ConnectBtn_clicked();

    //void on_SendBtn_clicked();

    void on_ClearBtn_clicked();

    void on_CheckBtn_clicked();

private:
    Ui::Widget *ui;
    QTcpSocket *myTcpSocket;
    bool pushBtnFlag=false;
    QSerialPort *mySerial;
    QByteArray tempStr;//保存的从服务器获取的字节流
    QByteArray readComDataMsg;//保存从串口获取的字节流
};

#endif // WIDGET_H
main.cpp:

#include "widget.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Widget w;
    w.show();
    w.setWindowTitle("串口转Tcp工具");
    return a.exec();
}
widget.cpp:

#include "widget.h"
#include "ui_widget.h"
#include<QFile>
Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    //初始化端口
    InitPort();
    //初始化Tcp
    InitTcp();
   // ui->SendBtn->setEnabled(false);

}
Widget::~Widget()
{
    delete ui;
}
void Widget::InitTcp()
{
    myTcpSocket=new QTcpSocket(this);
    //由于connectToHost没有返回值,所以通过三个connect来判断和服务器的状态
    connect(myTcpSocket,SIGNAL(disconnected()),this,SLOT(doProssDisconnected()));
    connect(myTcpSocket,SIGNAL(connected()),this,SLOT(doProssConnected()));
    connect(myTcpSocket,SIGNAL(readyRead()),this,SLOT(doProcessReadyRead()));
}
//点击打开Tcp连接服务器
void Widget::on_pushButton_clicked()
{
    if(ui->pushButton->text()=="打开TCP")
    {
        QString ServceIp=ui->lineEdit_IP->text();
        QString ServcePort=ui->lineEdit_Port->text();
        myTcpSocket->connectToHost(QHostAddress(ServceIp),ServcePort.toUInt());
        ui->pushButton->setText("关闭TCP");
    }
    else if(ui->pushButton->text()=="关闭TCP")
    {
      myTcpSocket->close();
      ui->pushButton->setText("打开TCP");
      ui->textEdit_Client->append("TCP关闭成功!");
    }
}
void Widget::doProssConnected()
{
    QString str="打开服务器成功";
    ui->textEdit_Client->append(str);
}
void Widget::doProcessReadyRead()
{
    QTcpSocket *myTcpSocket=(QTcpSocket *)this->sender();
    //读取服务器向缓冲区的存储数据
    while (!myTcpSocket->atEnd())
    {
     tempStr= myTcpSocket->readAll();
   // tempStr=QString::fromLocal8Bit(tempMsg);
     qDebug()<<"服务器向缓冲区存储字节数据";
    }
    //处理向串口写入数据
    doProcessWrite();
    writeByteArray();

}
void Widget::doProssDisconnected()
{
//    QString msg="服务器断开";
//    ui->textEdit_Client->append(msg);
    ui->pushButton->setText("关闭TCP");

}
//向服务器写入数据。
void Widget::doProcessTcpWrite()
{
    if(!readComDataMsg.isEmpty())
    {
     int ret= myTcpSocket->write(readComDataMsg);
     readComDataMsg.clear();
       if(ret<0)
       {
           return;
       }
    }
    else{
         qDebug()<<"向服务器写入数据失败";
    }

}


//........................................................................
//.............................串口.......................................
void Widget::InitPort()
{
    mySerial = new QSerialPort(this);
    foreach (const QSerialPortInfo&info,QSerialPortInfo::availablePorts())
    {

        QSerialPort serial;
        serial.setPort(info);
        //串口每打开一次就要close一次,不然下次打不开。
        qDebug()<<"#######";
        if(serial.open(QIODevice::ReadWrite))
        {
        ui->ComBC->addItem(info.portName());
        serial.close();
        }
    }

    QStringList baudList;//波特率
    baudList<<"115200"<<"57600"<<"38400"<<"19200"<<"9600"<< "4800"<<"2400"<<"1200";
    ui->BaudCB->addItems(baudList);
    QStringList dataBitsList;
    dataBitsList<<"8"<<"7"<<"6"<<"5";
    ui->DataBitsCB->addItems(dataBitsList);
    QStringList parityList;
    parityList<<"无校验"<<"奇校验"<<"偶校验";
    ui->ParityCB->addItems(parityList);
    QStringList stopBitsList;
    stopBitsList<<"1"<<"1.5"<<"2";
    ui->StopCB->addItems(stopBitsList);
    QStringList setFlowCtrl;
    setFlowCtrl<<"off"<<"RTS/CTS"<<"XON/XOFF";
    ui->FlowsCB->addItems(setFlowCtrl);
    //ui->SendBtn->setEnabled(false);
}
void Widget::on_ConnectBtn_clicked()
{
    if(ui->ConnectBtn->text()=="串口连接")
      {
       mySerial->setPortName(ui->ComBC->currentText());
       qDebug()<<"***********";
       bool openSerial= mySerial->open(QIODevice::ReadWrite);

       if(openSerial)
        {
            //设置波特率
            if ( ui->BaudCB->currentText()=="115200")
            {
                mySerial->setBaudRate(QSerialPort::Baud115200);
            }
                //qDebug()<<"115200";
            else if ( ui->BaudCB->currentText()==" 9600")
            {
                 mySerial->setBaudRate(QSerialPort::Baud9600);
            }
            else if ( ui->BaudCB->currentText()==" 1200")
            {
                mySerial->setBaudRate(QSerialPort::Baud1200);
               }
             else if ( ui->BaudCB->currentText()==" 2400")
              {
                 mySerial->setBaudRate(QSerialPort::Baud2400);
                }
            else if ( ui->BaudCB->currentText()==" 4800")
            {

                mySerial->setBaudRate(QSerialPort::Baud4800);
             }
             else if ( ui->BaudCB->currentText()==" 19200")
            {
                 mySerial->setBaudRate(QSerialPort::Baud19200);
             }
            else if ( ui->BaudCB->currentText()==" 38400")
            {
                mySerial->setBaudRate(QSerialPort::Baud38400);
             }
            else if ( ui->BaudCB->currentText()==" 57600")
            {
                mySerial->setBaudRate(QSerialPort::Baud57600);
             }

            //设置数据位
            if (ui->DataBitsCB->currentText()=="8")

                  mySerial->setDataBits(QSerialPort::Data8);
            else if (ui->DataBitsCB->currentText()=="7")

                 mySerial->setDataBits(QSerialPort::Data7);
            else if (ui->DataBitsCB->currentText()=="6")

                 mySerial->setDataBits(QSerialPort::Data6);
            else if (ui->DataBitsCB->currentText()=="5")

                mySerial->setDataBits(QSerialPort::Data5);

            //设置校验位
            if (ui->ParityCB->currentText()=="0")
                 mySerial->setParity(QSerialPort::NoParity);
            else  if (ui->ParityCB->currentText()=="2")
                mySerial->setParity(QSerialPort::EvenParity);
            else if (ui->ParityCB->currentText()=="3")
                 mySerial->setParity(QSerialPort::OddParity);

            //停止位
            if (ui->StopCB->currentText()=="1")

                mySerial->setStopBits(QSerialPort::OneStop);
             else if (ui->StopCB->currentText()=="3")
                  mySerial->setStopBits(QSerialPort::OneAndHalfStop);
             else if (ui->StopCB->currentText()=="2")
                mySerial->setStopBits(QSerialPort::TwoStop);
            //流控制
            if (ui->FlowsCB->currentText()=="0")

                 mySerial->setFlowControl(QSerialPort::NoFlowControl);
            else if (ui->FlowsCB->currentText()=="1")
                mySerial->setFlowControl(QSerialPort::HardwareControl);
            else if (ui->FlowsCB->currentText()=="2")
                mySerial->setFlowControl(QSerialPort::SoftwareControl);
        }
       connect(mySerial,SIGNAL(readyRead()),this,SLOT(doProcessSerialRead()));
       QString msg1="log gpgga ontime 5\r\n";
       QString msg2="interfacemode auto auto on\r\n";
       //mySerial->write(msg1+msg2);
       mySerial->write(msg1.toLatin1());
       mySerial->write(msg2.toLatin1());
        ui->ConnectBtn->setText("串口关闭");
        ui->BaudCB->setEnabled(false);
        ui->DataBitsCB->setEnabled(false);
        ui->FlowsCB->setEnabled(false);
        ui->ParityCB->setEnabled(false);
        ui->StopCB->setEnabled(false);
        ui->ComBC->setEnabled(false);
      }
    else if(ui->ConnectBtn->text()=="串口关闭")
        {
          mySerial->close();
          ui->ConnectBtn->setText("串口连接");
          ui->BaudCB->setEnabled(true);
          ui->DataBitsCB->setEnabled(true);
          ui->FlowsCB->setEnabled(true);
          ui->ParityCB->setEnabled(true);
          ui->StopCB->setEnabled(true);
          ui->ComBC->setEnabled(true);
        }
}
//读取串口的数据
void Widget::doProcessSerialRead()
{
     readComDataMsg = mySerial->readAll();

     QString str=QString::fromLocal8Bit(readComDataMsg);
     QString strTemp=str.replace(QString("\r\n"),QString(" "));
     ui->textEdit_Servce->append(strTemp);
     //ui->textEdit_Servce->setText(readComDataMsg);
//     }
       doProcessTcpWrite();
      // readComData.clear();

}
//void Widget::on_SendBtn_clicked()
//{
//    //QString serialTemp=ui->textEdit_Client->toPlainText();
//   QString serialTemp="log versionb ontime 1\r\n";
//    //QString serialTemp="log gpgga ontime 1 \r\n";
//    //qDebug()<<serialTemp;
//   int ret= mySerial->write(serialTemp.toLatin1());
//   qDebug()<<ret;
//   qDebug()<<serialTemp;
//   if(ret<0)
//   {
//       qDebug()<<ret;
//   }
//}
void Widget::on_ClearBtn_clicked()
{
    ui->textEdit_Servce->clear();
    ui->textEdit_Client->clear();
}
//向串口写入数据
void Widget::doProcessWrite()
{

    if(!tempStr.isEmpty())
    {
        qDebug()<<"向串口写入数据正确";
//        char *ch=tempStr.data();
//    int ret= mySerial->write(ch);
      mySerial->write(tempStr);
      int byteLen= mySerial->bytesToWrite();
      qDebug()<<"写数据的大小:"<<byteLen;
      //qDebug()<<"这个是向串口写入数据:"<<mySerial;
//       if(tempStr.size()<0)
//       {
//           return;
//       }
    }
    else{
         qDebug()<<"向串口写入数据错误";
        return;
    }
    QString str=QString::fromLocal8Bit(tempStr);
    QString strReplace= str.replace(QString("\r\n"),QString(""));
    if(!strReplace.isEmpty())
    {
    ui->textEdit_Client->append(strReplace);
    //qDebug()<<strReplace;
    }
    else {
         return;
    }
    //tempStr.clear();
}

void Widget::writeByteArray()
{
    //写二进制文件
       QFile file("D://testBase.txt");
       if(!file.open(QIODevice::WriteOnly | QIODevice::Append))
       {
           qDebug() << "Can't open file for writing";
       }
//       QDataStream out(&file);
//       out.setVersion(QDataStream::Qt_5_6);
//       out << tempStr;
       qDebug()<<"向文件写入数据正确";
       file.write(tempStr);
       file.close();
       tempStr.clear();
}
//刷新串口
void Widget::on_CheckBtn_clicked()
{
    ui->ComBC->clear();
    foreach (const QSerialPortInfo&info,QSerialPortInfo::availablePorts())
    {

        QSerialPort serial;
        serial.setPort(info);
        if(serial.open(QIODevice::ReadWrite))
        {
        ui->ComBC->addItem(info.portName());
        serial.close();
        }
    }
}

代码思路梳理:从服务器获取字节流保存在tempStr全局变量中,然后写入到串口,从串口中获取的数据保存在readComDataMsg全局变量中,写入到服务器和在界面上显示。

继续优化的问题:

1.进行大量串口数据的转发,运行一天软件正常运行,数据转发正常,可以优化成多线程间通信,拓展知识。

2.doProcessWrite()/doProcessTcpWrite()这两个函数时直接调用的,应该使用Qt的风格,用emit发送信号,使用信号和槽函数模式。

       欢迎有问题的小伙伴或者对代码有见解的,多多交流哦。

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

串口通信通过Tcp传到服务器实现串口数据转发 的相关文章

  • Qt中正确的线程方式

    我的图像加载非常耗时 图像很大 并且在加载时也完成了一些操作 我不想阻止应用程序 GUI 我的想法是在另一个线程中加载图像 发出图像已加载的信号 然后用该图像重绘视图 我的做法 void Window loadImage ImageLoad
  • QWidget::showMinimized() 不起作用

    在 Ubuntu 13 04 上 如果使用QWidget showMinimized 为了最小化窗口 我发现通过单击系统任务栏上的应用程序图标恢复它后 调用QWidget showMinimized 无法工作 connect minimum
  • 如何将枚举类传递给 QML?

    我正在学习QML with Qt并在通过时遇到一些麻烦enum class to qml 当我使用信号时int参数 一切正常 代码运行完美 But 麻烦就在这里 如果我使用信号与一些enum class我有参数undefinedqml 信号
  • Qt 图表和数据可视化小部件

    我已经安装了 Qt 5 7 来尝试 Qt 图表和 Qt 数据可视化 但我在 Qt Designer 和 Qt Creator 中都找不到新的小部件 有什么建议我应该做什么才能让新的小部件出现在设计器中 我今天遇到了完全相同的问题 默认情况下
  • 如何在 Qt-Creator 中添加自定义构建步骤?

    构建我的应用程序后 我想将其复制到特定目录 在 Windows 7 上 自定义构建步骤 cmd exe c k copy MyPlugin dll HostApp Debug plugins 但我有错误 Can t run process
  • QObject多重继承

    我正在尝试在 C Qt 类中使用 mix 来提供一大堆具有通用接口的小部件 该接口是以这样的方式定义的 如果它被定义为其他小部件类的基类 那么小部件本身将具有这些信号 class SignalInterface public QObject
  • QPainterPath::arcTo 上的角度如何解释?

    我正在开发图形编辑器的功能 在其中编辑弧线 当形状是椭圆形时 QPainterPath arcTo 的行为并不像我预期的那样 当它是一个圆圈时 它会按预期工作 下面的两张图片显示了结果 在第一种情况下 我创建了一个圆 然后将其转换为初始起始
  • QFileDialog 作为 TableView 的编辑器:如何获取结果?

    我正在使用一个QFileDialog作为某些专栏的编辑QTableView 这基本上有效 对一些焦点问题取模 请参阅here https stackoverflow com questions 22854242 qfiledialog as
  • 无法将 [未定义] 分配给 QColor

    我正在使用 Qt 5 11 构建 运行代码 代码中有QML风格如下 Button style delegate Component id enabledButtonStyle ButtonStyle padding top 0 paddin
  • 运行最新版本时没有“最新”消息?

    我正在尝试使用Sparkle https sparkle project org与 Qt Go 的绑定 https github com therecipe qt app 闪光 m import
  • qt 如何知道按钮被点击?

    我正在尝试编写一个程序 用声音进行一些操作 我的问题是我有 3 个播放按钮和 3 个标签 我希望无论我单击 播放 按钮 都应该播放按钮附近标签中名称的声音 我有一个没有任何参数的播放插槽 那么 如何分别连接到每个播放按钮和每个标签呢 实际上
  • QGraphicsSimpleTextItem“无效使用不完整类型”

    我的代码如下 指针部件 h QGraphicsSimpleTextItem text 指针控件 cpp void PointerWidget placeNumbers float spacing int currentTickNumber
  • MapItemView 在 dataChanged 信号后不会更新

    我正在使用 QMLMapItemView使用 C 的组件QAbstractListModel基于模型 这MapItemView当模型重置时 或者每当添加新项目或删除现有项目时 工作正常 但是 那MapItemView不反映对已添加项目的更改
  • Qt 和 MOC 的困境与简单的制作

    我想这更像是一个 GNU Make 问题 而不是 Qt 和 moc 但这里是 我有一个包含多个目录Q OBJECTS 我有一些简单的代码 它收集所有这些 例如 MOCS shell grep l Q OBJECT HEADERS Assum
  • Qt:关闭期间线程仍在运行时 qthread 被销毁

    我有一堂课 class centralDataPool public QObject Q OBJECT public centralDataPool QObject parent 0 centralDataPool commMonitor
  • Qt QML MenuItem iconSource不显示

    我有一个非常简单的设置只是为了说明问题 import QtQuick Controls 1 4 import QtQuick Window 2 2 ApplicationWindow visible true width 640 heigh
  • 在 Windows 上静默安装 Qt55 Enterprise

    编辑 在 Qt 支持的帮助下 我已经解决了如何自动化 Qt 企业安装程序的这两个部分 下面是脚本调用 我正在尝试在 Windows 8 1 和 Windows 10 上静默安装 Qt 5 5 1 Enterprise 使用 script 开
  • QWebView等待加载

    bool MainWindow waitForLoad QWebView view QEventLoop loopLoad QTimer timer QObject connect view SIGNAL loadFinished bool
  • QGraphicsView 在完整布局中未最大化

    I have following GUI having four QGraphicView objects 正如您在每个视图下看到的那样 它有四个工具按钮 为了最大化视图 我连接了工具按钮的信号来隐藏其他三个视图的插槽 并将大小策略设置为扩
  • Qt:关闭模式对话框关闭程序

    在我的 Qt 程序中 我有 2 个窗口 主窗口和子窗口 在程序中 一次仅显示这些窗口之一 主窗口有一个插槽 用于创建模式对话框 现在 假设子窗口中单击按钮的信号被发送到该插槽 在这种情况下 主窗口隐藏 子窗口可见 对话框显示得很好 但是当对

随机推荐