Qt应用软件【串口篇】串口通信

news2024/11/17 0:17:56

文章目录

  • 1.串口概述
  • 2.串口传输数据的基本原理
    • 电信号的传输过程
  • 3.串口的几个概念
    • 数据位(Data Bits)
    • 奇偶校验位(Parity Bit)
    • 停止位(Stop Bits)
    • 流控制(Flow Control)
    • 波特率(Baud Rate)
    • 串口名(Port Name)
  • 4.CH340串口接线方法
    • 简述
    • 接线步骤
      • 接线步骤
      • 注意事项
  • 5.Qt串口API
    • QSerialPort 类
    • QSerialPortInfo 类
    • 信号和槽
  • 6.串口通信代码示例
  • 7.串口编程技巧

1.串口概述

串口通信是计算机与外部设备进行数据交换的一种基础通信方式。它使用串行通信协议,其中数据按位顺序传输,通常用于较低速率的通信。串口通信广泛应用于各种设备,如打印机、鼠标、调制解调器等

串口通信通常使用 RS-232、RS-422 或 RS-485 等标准。其中,RS-232 是最常用的串口通信标准,它在计算机、通信设备和其他类型的设备之间广泛使用。

2.串口传输数据的基本原理

  1. 数字信号到电信号的转换:在串口通信中,数据首先以二进制形式存在。这些二进制数据(0和1)被转换为电信号。在 RS-232 标准中,一般情况下,负电压代表二进制位“1”(逻辑高),正电压代表二进制位“0”(逻辑低)。电压的具体值可以根据标准的不同而有所变化。
  2. 串行数据传输:与并行传输不同,串行传输是指数据位按顺序一个接一个地通过单个线路传输。这意味着数据位(比特)是连续发送的,而不是同时发送。
  3. 起始位和停止位:每个数据包通常以一个起始位开始,该起始位通知接收设备一个新字节的开始。数据位随后按顺序发送。最后,数据包以一个或多个停止位结束,表示数据字节的结束。
  4. 奇偶校验位(可选):为了错误检测,有时会在数据位之后添加一个奇偶校验位。这个位可以用来检查数据在传输过程中是否出现错误

电信号的传输过程

  1. 线路驱动器:串口通信中的发送设备包含一个线路驱动器,该驱动器负责将二进制数据转换为适合在物理介质上传输的电信号。
  2. 传输介质:电信号通过串行接口的物理线路传输,例如使用双绞线。
  3. 线路接收器:接收设备包含一个线路接收器,负责将收到的电信号转换回二进制数据。
  4. 同步:为了正确地解释接收到的数据,接收设备需要与发送设备同步。这通常通过观察起始位和停止位来实现,以确定数据位的边界
  • 时序图
二进制数据 转换器(驱动器) 物理线路(电缆) 接收器 解码数据 数据 (0和1) 将二进制转换为电信号 电信号 通过电缆传输 电信号 数据 (0和1) 将电信号转换为二进制 二进制数据 转换器(驱动器) 物理线路(电缆) 接收器 解码数据
  • 串口连线图
2 (RXD)
3 (TXD)
5 (GND)
2 (TXD)
3 (RXD)
7 (GND)
DB9 Connector
DB25 Connector
  • 对于 DB9 连接器:
    • 引脚 2 是接收数据(RXD)
    • 引脚 3 是发送数据(TXD)
    • 引脚 5 是地线(GND)
  • 对于 DB25 连接器:
    • 引脚 2 是发送数据(TXD)
    • 引脚 3 是接收数据(RXD)
    • 引脚 7 是地线(GND)

3.串口的几个概念

数据位(Data Bits)

数据位是串口通信中每个数据包的大小,表示每个数据字节中用于承载信息的位数。常见的数据位设置有 7 位或 8 位。例如,8 数据位意味着每个数据字节包含 8 个位。

奇偶校验位(Parity Bit)

奇偶校验位是一种错误检测机制。它在每个数据字节后添加一个额外的校验位,用于检测数据传输过程中的单比特错误。奇偶校验可以是无校验、奇校验或偶校验。

停止位(Stop Bits)

停止位用于标记每个数据包的结束。它在每个数据字节后添加一定数量的额外位,以指示数据包的终止。常见的停止位设置有 1 位、1.5 位和 2 位。

流控制(Flow Control)

流控制是用于管理数据传输速率和确保数据完整性的一种机制。它可以防止发送方在接收方来得及处理之前发送过多数据。流控制可以是无流控制、硬件流控制(RTS/CTS)或软件流控制(XON/XOFF)。

波特率(Baud Rate)

波特率是衡量串口通信速度的单位,表示每秒钟可以传输多少比特(位)的数据。常见的波特率设置有 9600、19200、38400、57600 和 115200 等。选择合适的波特率取决于设备的能力和通信距离。

串口名(Port Name)

串口名是用于标识计算机上特定串口的名称。在 Windows 系统中,串口名通常是以 “COM” 开头的,如 “COM1”、“COM2” 等。在 Unix/Linux 系统中,串口名通常以 “/dev/tty” 开头,如 “/dev/ttyS0”、“/dev/ttyUSB0” 等。

这些参数共同定义了串口通信的基本特性,包括数据的格式、速率和传输控制方式。正确配置这些参数对于确保串口通信的可靠性和效率至关重要。

4.CH340串口接线方法

简述

CH340 是一种常见的 USB 转串口芯片,广泛用于提供 USB 到串行通信的接口

  • 安装官方驱动

http://sparks.gogo.co.nz/ch340.html

  • USB转串口CH340接线

在这里插入图片描述

  • VCC:电源输入,通常为 3.3V 或 5V。
    TXD (Transmit Data):数据发送线,用于发送数据到另一设备
    RXD (Receive Data):数据接收线,用于接收来自另一设备的数据。
    GND (Ground):地线,用于电气接地

接线步骤

USB 转 TTL 串口模块(如基于 CH340 芯片的模块)可以用来将 USB 接口转换为 TTL 串口(即逻辑电平串口)。这种模块通常用于连接电脑的 USB 端口和使用 TTL 电平通信的设备,如微控制器或其他串口设备。以下是 USB 转 TTL 串口模块(CH340)的一般接线方法:

USB 转 TTL 模块(CH340)的典型管脚

  1. VCC:提供电源(通常为 3.3V 或 5V)。
  2. GND:地线,用于接地。
  3. TXD:发送数据(Transmit Data),CH340 的 TXD 连接到目标设备的 RXD。
  4. RXD:接收数据(Receive Data),CH340 的 RXD 连接到目标设备的 TXD。
  5. DTRRTS 等(如果有):用于流控制,根据需要连接。

接线步骤

  1. 连接 VCC 和 GND

    • 将 CH340 模块的 VCC 连接到目标设备的 VCC(注意电压匹配,通常为 3.3V 或 5V)。
    • 将 CH340 模块的 GND 连接到目标设备的 GND。
  2. 连接 TXD 和 RXD

    • 将 CH340 模块的 TXD(发送数据)连接到目标设备的 RXD(接收数据)。
    • 将 CH340 模块的 RXD(接收数据)连接到目标设备的 TXD(发送数据)。

    这种交叉连接确保一方的发送线连接到另一方的接收线。

  3. 连接流控制引脚(可选)

    • 如果您的应用需要流控制,可以连接 CH340 模块的 DTR、RTS 等管脚到目标设备相应的引脚。

注意事项

  • 确保电源电压匹配,避免对设备造成损害。
  • 确认目标设备的 TTL 电平是否与 CH340 模块兼容(通常为 3.3V 或 5V TTL 电平)。
  • 如果设备需要特殊的初始化序列(例如进入引导模式),可能需要特别处理 DTR 或 RTS 等控制信号。

在连接之前,请检查目标设备的文档以确认正确的接线方法。每个设备的接线要求可能略有不同。

5.Qt串口API

QSerialPort 类

QSerialPort(QObject *parent = nullptr) // 构造函数,创建一个 QSerialPort 对象。
void setPortName(const QString &name) // 设置串口的端口名。
QString portName() const // 获取当前串口的端口名。
void setBaudRate(qint32 baudRate, QSerialPort::Directions directions = AllDirections) // 设置波特率。
qint32 baudRate(QSerialPort::Directions directions = AllDirections) const // 获取当前波特率。
void setDataBits(QSerialPort::DataBits dataBits) // 设置数据位。
QSerialPort::DataBits dataBits() const // 获取当前设置的数据位。
void setParity(QSerialPort::Parity parity) // 设置奇偶校验位。
QSerialPort::Parity parity() const // 获取当前奇偶校验设置。
void setStopBits(QSerialPort::StopBits stopBits) // 设置停止位。
QSerialPort::StopBits stopBits() const // 获取当前停止位设置。
void setFlowControl(QSerialPort::FlowControl flowControl) // 设置流控制。
QSerialPort::FlowControl flowControl() const // 获取当前流控制设置。
bool open(QIODevice::OpenMode mode) // 打开串口,设置打开模式(读、写、读写)。
void close() // 关闭串口。
bool isOpen() const // 检查串口是否打开。
bool isReadable() const // 检查串口是否可读。
bool isWritable() const // 检查串口是否可写。
bool writeData(const char *data, qint64 len) // 向串口写入数据。
bool readData(char *data, qint64 maxSize) // 从串口读取数据。
QByteArray readAll() // 读取串口接收到的所有数据。
qint64 bytesAvailable() const // 获取可读取的字节数。
bool waitForReadyRead(int msecs) // 等待数据可读,带有超时。
bool waitForBytesWritten(int msecs) // 等待数据写入完成,带有超时。
QSerialPort::SerialPortError error() const // 获取当前的错误状态。
void clearError() // 清除当前的错误状态。

QSerialPortInfo 类

static QList<QSerialPortInfo> availablePorts() // 获取系统中所有可用的串口列表。
QString portName() const // 获取串口的端口名。
QString description() const // 获取串口的描述信息。
QString manufacturer() const // 获取串口的制造商信息。
QString serialNumber() const // 获取串口的序列号。
QString systemLocation() const // 获取串口在系统中的位置。
qint16 vendorIdentifier() const // 获取串口的供应商标识符。
qint16 productIdentifier() const // 获取串口的产品标识符。
bool hasVendorIdentifier() const // 检查是否有供应商标识符。
bool hasProductIdentifier() const // 检查是否有产品标识符。
bool isBusy() const // 检查串口是否正在使用中。

信号和槽

void readyRead() // 当串口有数据可读时发出的信号。
void bytesWritten(qint64 bytes) // 当串口有数据写入时发出的信号。
void errorOccurred(QSerialPort::SerialPortError error) // 当串口发生错误时发出的信号。

这些函数提供了在 Qt 中进行串口通信所需的基本功能,包括配置串口参数、打开/关闭串口、读写数据以及处理错误和事件。

6.串口通信代码示例

#include <QApplication>
#include <QWidget>
#include <QSerialPort>
#include <QSerialPortInfo>
#include <QPushButton>
#include <QTextEdit>
#include <QVBoxLayout>
#include <QDebug>
#include <QLabel>
#include <QComboBox>
#include <QMessageBox>

class SerialPortWidget : public QWidget
{
    Q_OBJECT

public:
    SerialPortWidget(QWidget *parent = nullptr) : QWidget(parent)
    {
        serial = new QSerialPort(this);

        auto layout = new QVBoxLayout(this);
        auto sendButton = new QPushButton(tr("Send"), this);
        auto refreshButton = new QPushButton(tr("Refresh Ports"), this);
        comboBox = new QComboBox(this);
        textEdit = new QTextEdit(this);
        textEdit->setReadOnly(true);

        layout->addWidget(new QLabel(tr("Select Serial Port:")));
        layout->addWidget(comboBox);
        layout->addWidget(sendButton);
        layout->addWidget(refreshButton);
        layout->addWidget(textEdit);

        refreshSerialPorts();

        connect(sendButton, &QPushButton::clicked, this, &SerialPortWidget::sendData);
        connect(refreshButton, &QPushButton::clicked, this, &SerialPortWidget::refreshSerialPorts);
        //串口可以读取事件
        connect(serial, &QSerialPort::readyRead, this, &SerialPortWidget::readData);
        //串口发生错误
        connect(serial, &QSerialPort::errorOccurred, this, &SerialPortWidget::handleError);
    }

private slots:
    void sendData()
    {
        QString data = "Hello, Serial Port!";
        serial->write(data.toLocal8Bit());
    }

    void readData()
    {
        QByteArray data = serial->readAll();
        textEdit->append(QString(data));
    }

    void handleError(QSerialPort::SerialPortError error)
    {
        if (error == QSerialPort::ResourceError) {
            QMessageBox::critical(this, tr("Critical Error"), serial->errorString());
            serial->close();
        }
    }

    void refreshSerialPorts()
    {
        comboBox->clear();
        //获取所有可用串口
        const auto infos = QSerialPortInfo::availablePorts();
        for (const QSerialPortInfo &info : infos) {
            comboBox->addItem(info.portName());
        }
    }

    void openSerialPort()
    {
    	//打开串口名
        serial->setPortName(comboBox->currentText());
        //波特率
        serial->setBaudRate(QSerialPort::Baud9600);
        //数据位
        serial->setDataBits(QSerialPort::Data8);
        //奇偶校验位
        serial->setParity(QSerialPort::NoParity);
        //停止位
        serial->setStopBits(QSerialPort::OneStop);
        //流控制
        serial->setFlowControl(QSerialPort::NoFlowControl);

        if (!serial->open(QIODevice::ReadWrite)) {
            QMessageBox::critical(this, tr("Error"), serial->errorString());
            return;
        }
    }

    void closeSerialPort()
    {
        if (serial->isOpen())
            serial->close();
    }

private:
    QSerialPort *serial;
    QTextEdit *textEdit;
    QComboBox *comboBox;
};

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    SerialPortWidget widget;
    widget.show();
    return app.exec();
}

7.串口编程技巧

  1. 串口枚举和检测

    • 在开发串口应用程序时,首先需要检测和枚举系统中可用的串口。QSerialPortInfo 类可以用来获取系统中所有可用串口的详细信息,包括端口名、描述、制造商等。
  2. 异步通信

    • Qt 串口模块支持异步通信,即你可以在不阻塞主线程的情况下进行读写操作。通过使用信号和槽机制,你可以实现非阻塞式的数据读写。例如,readyRead() 信号可用于在有数据可读时通知应用程序。
  3. 线程和串口通信

    • 在某些情况下,可能需要在单独的线程中处理串口通信,以避免主界面线程因长时间的 I/O 操作而变得不响应。在这种情况下,你需要确保线程之间正确地共享串口资源并同步数据。
  4. 错误处理

    • 处理串口通信中可能出现的错误是非常重要的。QSerialPort 提供了 errorOccurred() 信号和 error() 方法来报告和查询错误状态。
  5. 信号调制和解调

    • 通常,串口通信涉及信号的调制和解调,特别是在涉及长距离或无线传输时。虽然这些功能超出了 Qt 串口模块的直接支持范围,但了解这些概念对于设计健壮的通信系统是有益的。
  6. 跨平台兼容性

    • Qt 串口模块是跨平台的,这意味着你可以在 Windows、Linux、macOS 等操作系统上以相同的方式使用这些 API。
  7. 测试和调试

    • 在开发串口通信应用程序时,测试和调试是非常重要的。你可能需要使用串口调试工具来模拟串口设备,或者捕获和分析串口数据。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1418241.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

找到满意的北京软件外包公司

寻得一家满意的软件外包开发公司&#xff0c;需明确自身需求&#xff0c;细心调研&#xff0c;筛选比较&#xff0c;这样方能找到技术实力雄厚、服务贴心的合作伙伴&#xff0c;助力企业数字化转型之路。要找到一家满意的软件外包开发公司&#xff0c;需要遵循以下几个步骤&…

UDF学习(七)非稳态宏和对流宏及UDS_DIFFUCITY宏

非稳态宏和对流宏—FLUENT UDF-DEFINE_UDS_UNSTEADY宏 非稳态如何挂载 UDF_DEFINE_UDS_FLUX宏 对流项的宏&#xff0c;可以从help文件中直接用 FLUENT UDF-DEFINE_UDS_DIFFUCITY宏 定义了扩散系数 两个宏&#xff1a;DEFINE_ANISOTROPIC_DIFFUSITY宏和DEFINE_DIFFUSIVITY&a…

校园圈子论坛系统--APP小程序H5,前后端源码交付,支持二开!uniAPP+PHP书写!

随着移动互联网的快速发展&#xff0c;校园社交成为了大学生们日常生活中重要的一部分。为了方便校园内学生的交流和互动&#xff0c;校园社交小程序逐渐走入人们的视野。本文将探讨校园社交小程序的开发以及其带来的益处。 校园社交小程序的开发涉及许多技术和设计方面。首先&…

用友移动管理系统 DownloadServlet 任意文件读取漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…

MySQL:三大日志(binlog、redolog、undolog)

再了解三个日志前我们先了解一下MySQL的两层架构&#xff1a; Server 层负责建立连接、分析和执行 SQL。MySQL 大多数的核心功能模块都在这实现&#xff0c;主要包括连接器&#xff0c;查询缓存、解析器、预处理器、优化器、执行器等。另外&#xff0c;所有的内置函数和所有跨…

雅特力AT32 Workbench图形化代码生成工具,简化嵌入式开发利器

嵌入式系统应用市场广泛&#xff0c;早已遍及日常生活&#xff0c;随着产品需求复杂度的提升&#xff0c;32位MCU开发难度也随之增加&#xff0c;如何降低开发成本&#xff0c;缩短开发周期&#xff0c;是所有嵌入式开发人员的共同课题。 面对市场竞争日益加剧的情形&#xff…

JavaWeb创建详细

这个是自己总结的onenote 笔记 , 欢迎各位探讨指正

Lucene 查询原理

Lucene 查询原理 - 知乎 前言 Lucene 是一个基于 Java 的全文信息检索工具包&#xff0c;目前主流的搜索系统Elasticsearch和solr都是基于lucene的索引和搜索能力进行。想要理解搜索系统的实现原理&#xff0c;就需要深入lucene这一层&#xff0c;看看lucene是如何存储需要检…

移动端设计规范 - 文字使用规范

这是一篇关于移动端产品界面设计时&#xff0c;文字大小的使用规范&#xff0c;前端人员如果能了解一点的话&#xff0c;在实际开发中和设计沟通时&#xff0c;节省沟通成本&#xff0c;也能提高设计落地开发时的还原度。 关于 在做移动端产品设计时&#xff0c;有时候使用文字…

【JavaScript基础入门】06 JavaScript 数据类型

JavaScript 数据类型 目录 JavaScript 数据类型1. 数据类型1.1 数据类型简介1.2 简单数据类型1.2.1 简单数据类型&#xff08;基本数据类型&#xff09;1.2.2 数字型 N u m b e r \color{red}{Number} Number1.2.3 字符串型 S t r i n g \color{red}{String} String1.2.4 布尔…

介绍TCP/IP

TCP/IP&#xff08;传输控制协议/互联网协议&#xff09;是一种用于数据通信的基本通信协议&#xff0c;它是互联网的基础。TCP/IP指的是一组规则和过程&#xff0c;它规定了如何在网络上发送和接收数据。这个协议族由两个主要部分组成&#xff1a;传输控制协议&#xff08;TCP…

SpringBoot 源码解析 - 持续更新

开始 spring initilizer&#xff1a;根据依赖构建工具、springboot 版本等生成 Java 工程。手把手教你手写一个最简单的 Spring Boot Starter Starter 命名规则 Spring 官方定义的 Starter 通常命名遵循的格式为 spring-boot-starter-{name}&#xff0c;例如 spring-boot-star…

蓝牙----蓝牙GAP层

蓝牙协议栈----GAP GAP的角色连接过程连接参数 GAP&#xff1a;通用访问配置协议层 gap的角色发现的模式与过程连接模式与过程安全模式与过程 CC2640R2F的GAP层抽象 GAP的角色 Broadcaster 广播电台 -不可连接的广播者。Observer 观察者 -扫描广播者但无法启动连接。Periphe…

QA-GNN: 使用语言模型和知识图谱的推理问答

Abstract 使用预训练语言模型&#xff08;LMs&#xff09;和知识图谱&#xff08;KGs&#xff09;的知识回答问题的问题涉及两个挑战&#xff1a;在给定的问答上下文&#xff08;问题和答案选择&#xff09;中&#xff0c;方法需要&#xff08;i&#xff09;从大型知识图谱中识…

React一学就会(4): 强化练习二

书接上回&#xff0c;是不是感觉已经有点入门了。不过别急&#xff0c;码哥我准备了很多干货&#xff0c;等我们把这些基本几个章节的学完&#xff0c;码哥带着你一起装逼一起飞。我不大可能只是带着你们入门&#xff0c;那不是我的风格。码哥会教你如何开发一个完整中后台。前…

知识增强系列 ERNIE: Enhanced Representation through Knowledge Integration,论文解读

论文全称&#xff1a;通过知识集成增强语义表达 1. motivation ERNIE 目的在于通过知识屏蔽策略增强语言表示&#xff0c;其中屏蔽策略包括实体级屏蔽(Entity-level strategy)和短语级屏蔽(Phrase-level strategy)。 entity-level 策略通常会掩盖由多个单词组成的实体; Phrase…

智能加湿器数据分析:预计2025年市场规模将达到164.18亿元

随着经济的发展和人民生活水平的提高&#xff0c;人们对生活质量和健康的要求愈来愈高。空气加湿器慢慢的走进家庭当中&#xff0c;预计2023年中国线上超声波加湿器零售额同比下降4.9%;线上纯净型加湿器零售额同比增长44.8%。随着社会科技的不断进步和居民消费水平的不断提高&a…

【网络】WireShark过滤 | WireShark实现TCP三次握手和四次挥手

目录 一、开启WireShark的大门 1.1 WireShark简介 1.2 常用的Wireshark过滤方式 二、如何抓包搜索关键字 2.1 协议过滤 2.2 IP过滤 ​编辑 2.3 过滤端口 2.4 过滤MAC地址 2.5 过滤包长度 2.6 HTTP模式过滤 三、ARP协议分析 四、WireShark之ICMP协议 五、TCP三次握…

容器和虚拟机的对比

容器和虚拟机的对比 容器和虚拟机在与硬件和底层操作系统交互的方式上有所不同 虚拟化 使多个操作系统能够同时在一个硬件平台上运行。 使用虚拟机监控程序将硬件分为多个虚拟硬件系统&#xff0c;从而允许多个操作系统并行运行。 需要一个完整的操作系统环境来支持该应用。…