当前位置: 首页 > news >正文

QT技巧之快速搭建串口收发平台

       Qt 串口模块(QtSerialPort)是 Qt 框架中用于与串口设备通信的组件,提供了跨平台的串口访问能力。

QtSerialPort 模块包含两个核心类:

  1. QSerialPort

    • 提供对串口的配置和数据传输功能(打开 / 关闭串口、读写数据)。
    • 支持异步和同步通信模式。
  2. QSerialPortInfo

    • 用于枚举系统中可用的串口设备(如 COM1、/dev/ttyUSB0)。
    • 获取串口详细信息(描述、制造商、序列号等)。

一、环境配置

  1. 安装 Qt
    下载并安装 Qt SDK(包含 Qt Creator IDE),推荐 Qt 5.15 或更高版本。

  2. 启用串口模块
    在项目的.pro文件中添加:

    QT += serialport
    

  3. 包含必要头文件

    #include <QSerialPort>        // 串口类
    #include <QSerialPortInfo>    // 串口信息类
    

二、界面设计(UI 文件)

使用 Qt Designer 设计界面,通常包含:

  1. 串口设置区

    • 串口下拉框(选择 COM 口)
    • 波特率下拉框(如 9600、115200)
    • 数据位、停止位、校验位下拉框
    • 打开 / 关闭串口按钮
  2. 数据收发区

    • 接收文本框(显示接收到的数据)
    • 发送文本框(输入要发送的数据)
    • 发送按钮

 

三、核心代码实现

1. 初始化串口管理类
// mainwindow.h
#include <QMainWindow>
#include <QSerialPort>
#include <QSerialPortInfo>QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();private slots:void on_pushButton_open_clicked();  // 打开/关闭串口按钮槽函数void on_pushButton_send_clicked();  // 发送数据按钮槽函数void readData();                   // 读取串口数据的槽函数private:Ui::MainWindow *ui;QSerialPort *serialPort;           // 串口对象指针
};

 2. 构造函数初始化

// mainwindow.cpp
MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);serialPort = new QSerialPort(this);// 扫描并添加可用串口到下拉框foreach (const QSerialPortInfo &info, QSerialPortInfo::availablePorts()) {ui->comboBox_port->addItem(info.portName());}// 添加常用波特率选项ui->comboBox_baud->addItems({"9600", "115200", "19200", "38400"});// 设置默认选项ui->comboBox_baud->setCurrentText("115200");ui->comboBox_dataBits->setCurrentText("8");ui->comboBox_stopBits->setCurrentText("1");ui->comboBox_parity->setCurrentText("None");// 连接信号和槽connect(serialPort, &QSerialPort::readyRead, this, &MainWindow::readData);
}
3. 打开 / 关闭串口功能
void MainWindow::on_pushButton_open_clicked()
{if (serialPort->isOpen()) {// 关闭串口serialPort->close();ui->pushButton_open->setText("打开串口");ui->statusBar->showMessage("串口已关闭");} else {// 设置串口参数serialPort->setPortName(ui->comboBox_port->currentText());serialPort->setBaudRate(ui->comboBox_baud->currentText().toInt());// 设置数据位switch (ui->comboBox_dataBits->currentText().toInt()) {case 5: serialPort->setDataBits(QSerialPort::Data5); break;case 6: serialPort->setDataBits(QSerialPort::Data6); break;case 7: serialPort->setDataBits(QSerialPort::Data7); break;case 8: serialPort->setDataBits(QSerialPort::Data8); break;default: serialPort->setDataBits(QSerialPort::Data8); break;}// 设置停止位和校验位(类似逻辑)// ...// 尝试打开串口if (serialPort->open(QIODevice::ReadWrite)) {ui->pushButton_open->setText("关闭串口");ui->statusBar->showMessage("串口已打开");} else {ui->statusBar->showMessage("串口打开失败: " + serialPort->errorString());}}
}
4. 数据接收功能
void MainWindow::readData()
{QByteArray data = serialPort->readAll();// 显示接收到的数据ui->textEdit_receive->append(data);
}
5. 数据发送功能
void MainWindow::on_pushButton_send_clicked()
{if (serialPort->isOpen()) {QString sendData = ui->textEdit_send->toPlainText();serialPort->write(sendData.toUtf8());ui->statusBar->showMessage("数据已发送");} else {ui->statusBar->showMessage("请先打开串口");}
}

四、完整实现示例

以下是一个完整的串口收发平台代码:

// mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QSerialPort>
#include <QSerialPortInfo>QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();private slots:void on_pushButton_open_clicked();void on_pushButton_send_clicked();void readData();private:Ui::MainWindow *ui;QSerialPort *serialPort;
};
#endif // MAINWINDOW_H// mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);serialPort = new QSerialPort(this);// 扫描可用串口foreach (const QSerialPortInfo &info, QSerialPortInfo::availablePorts()) {ui->comboBox_port->addItem(info.portName());}// 设置波特率选项ui->comboBox_baud->addItems({"9600", "115200", "19200", "38400"});ui->comboBox_baud->setCurrentText("115200");// 设置数据位选项ui->comboBox_dataBits->addItems({"5", "6", "7", "8"});ui->comboBox_dataBits->setCurrentText("8");// 设置停止位选项ui->comboBox_stopBits->addItems({"1", "1.5", "2"});ui->comboBox_stopBits->setCurrentText("1");// 设置校验位选项ui->comboBox_parity->addItems({"None", "Even", "Odd", "Space", "Mark"});ui->comboBox_parity->setCurrentText("None");// 连接信号和槽connect(serialPort, &QSerialPort::readyRead, this, &MainWindow::readData);
}MainWindow::~MainWindow()
{if (serialPort->isOpen()) {serialPort->close();}delete ui;
}void MainWindow::on_pushButton_open_clicked()
{if (serialPort->isOpen()) {serialPort->close();ui->pushButton_open->setText("打开串口");ui->statusBar->showMessage("串口已关闭");} else {// 设置串口参数serialPort->setPortName(ui->comboBox_port->currentText());serialPort->setBaudRate(ui->comboBox_baud->currentText().toInt());// 设置数据位switch (ui->comboBox_dataBits->currentText().toInt()) {case 5: serialPort->setDataBits(QSerialPort::Data5); break;case 6: serialPort->setDataBits(QSerialPort::Data6); break;case 7: serialPort->setDataBits(QSerialPort::Data7); break;case 8: serialPort->setDataBits(QSerialPort::Data8); break;default: serialPort->setDataBits(QSerialPort::Data8); break;}// 设置停止位if (ui->comboBox_stopBits->currentText() == "1") {serialPort->setStopBits(QSerialPort::OneStop);} else if (ui->comboBox_stopBits->currentText() == "1.5") {serialPort->setStopBits(QSerialPort::OneAndHalfStop);} else if (ui->comboBox_stopBits->currentText() == "2") {serialPort->setStopBits(QSerialPort::TwoStop);}// 设置校验位if (ui->comboBox_parity->currentText() == "None") {serialPort->setParity(QSerialPort::NoParity);} else if (ui->comboBox_parity->currentText() == "Even") {serialPort->setParity(QSerialPort::EvenParity);} else if (ui->comboBox_parity->currentText() == "Odd") {serialPort->setParity(QSerialPort::OddParity);}// 打开串口if (serialPort->open(QIODevice::ReadWrite)) {ui->pushButton_open->setText("关闭串口");ui->statusBar->showMessage("串口已打开");} else {ui->statusBar->showMessage("串口打开失败: " + serialPort->errorString());}}
}void MainWindow::on_pushButton_send_clicked()
{if (serialPort->isOpen()) {QString sendData = ui->textEdit_send->toPlainText();serialPort->write(sendData.toUtf8());ui->statusBar->showMessage("数据已发送");} else {ui->statusBar->showMessage("请先打开串口");}
}void MainWindow::readData()
{QByteArray data = serialPort->readAll();ui->textEdit_receive->append(data);
}// main.cpp
#include "mainwindow.h"
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}

五、关键特性技巧

1. 异步通信
  • 通过信号槽机制处理数据接收(推荐方式):
    connect(&serial, &QSerialPort::readyRead, this, [&]() {// 处理接收到的数据
    });
    
2. 同步通信
  • 使用waitForReadyRead()waitForBytesWritten()等阻塞函数(不推荐在 UI 线程中使用):
    if (serial.waitForReadyRead(1000)) {QByteArray data = serial.readAll();
    }
    

3. 错误处理
  • 通过error()信号或errorString()获取错误信息:
    connect(&serial, &QSerialPort::errorOccurred, this, [](QSerialPort::SerialPortError error) {qDebug() << "串口错误:" << error;
    });
    

4. 超时设置
  • 可设置读写超时时间(仅对同步模式有效):
    serial.setReadBufferSize(1024);  // 设置接收缓冲区大小
    

六、注意事项

  1. 跨平台兼容性
    QtSerialPort 支持 Windows、Linux、macOS 等,但不同系统的串口名称格式不同(如 COM1、/dev/ttyUSB0)。

  2. 线程安全
    串口数据接收是异步的,在复杂应用中需注意线程安全问题(可使用信号槽机制)。

  3. 性能优化
    大量数据收发时,需考虑缓冲区处理和界面刷新频率,避免 UI 卡顿。

http://www.dtcms.com/a/282326.html

相关文章:

  • Qt中实现文件(文本文件)内容对比
  • Django基础(三)———模板
  • Python设计模式深度解析:装饰器模式(Decorator Pattern)完全指南
  • hadoop 集群问题处理
  • 肠道宏基因组数据分析流程
  • 肠道宏基因组数据分析流程要关注的参数和指标
  • STM32-RTC内部时钟
  • 图像质量评价(Image Quality Assessment,IQA)
  • 【unitrix】 6.1 类型化整数特征(t_int.rs)
  • 深入理解-Java-线程池:原理、动态调整与监控实践
  • 牛市看涨期权的价差策略是什么?
  • mongoDB初始化项目简单操作示例
  • YAML 自动化用例中 GET vs POST 请求的参数写法差异
  • 部分排序算法的Java模拟实现(复习向,非0基础)
  • PostgreSQL数据库集群如何进行自动化性能监测?
  • HTML5》》template
  • (LeetCode 面试经典 150 题) 205. 同构字符串 (哈希表)
  • 针对 Python、Java、Go 的依赖树检测实现方案,包含漏洞扫描和依赖关系分析的核心代码与工具链
  • Chrome紧急更新,谷歌修复正遭活跃利用的关键零日漏洞
  • Java运维之Tomcat升级
  • 【c++深入系列】:万字详解list(附模拟实现的list源码)
  • Android 高通平台修改音频参数效果文件-优化音频效果
  • 如何使用 OpenCV 打开指定摄像头
  • 微服务变更?自动化测试利器Parasoft SOAtest修复快、准、稳!
  • 【微服务】Ocelot微服务网关
  • RL-马尔科夫过程、动态规划
  • 042_封装的实现(属性私有化 / 方法公开)
  • 网络基础10 长途互联--WAN广域网技术
  • fastadmin中ajax弹窗修改文字为英文
  • Taro.getRandomValues() 用法详解