Qt串口通信开发之QSerialPort模块详细使用方法与实例

Qt串口通信基础及名词说明

串口通信(Serial Communications)的概念非常简单,串口按位(bit)发送和接收字节。尽管比按字节(byte)的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据。它很简单并且能够实现远距离通信。比如IEEE488定义并行通行状态时,规定设备线总长不得超过20米,并且任意两个设备间的长度不得超过2米;

而对于串口而言,长度可达1200米。典型地,串口用于ASCII码字符的传输。通信使用3根线完成,分别是地线、发送、接收。由于串口通信是异步的,端口能够在一根线上发送数据同时在另一根线上接收数据。其他线用于握手,但不是必须的。串口通信最重要的参数是波特率、数据位、停止位和奇偶校验。对于两个进行通信的端口,这些参数必须匹配。

波特率:

这是一个衡量符号传输速率的参数。指的是信号被调制以后在单位时间内的变化,即单位时间内载波参数变化的次数,如每秒钟传送240个字符,而每个字符格式包含10位(1个起始位,1个停止位,8个数据位),这时的波特率为240Bd,比特率为10位*240个/秒=2400bps。一般调制速率大于波特率,比如曼彻斯特编码)。通常电话线的波特率为14400,28800和36600。波特率可以远远大于这些值,但是波特率和距离成反比。高波特率常常用于放置的很近的仪器间的通信,典型的例子就是GPIB设备的通信。

数据位:

这是衡量通信中实际数据位的参数。当计算机发送一个信息包,实际的数据往往不会是8位的,标准的值是6、7和8位。如何设置取决于你想传送的信息。比如,标准的ASCII码是0~127(7位)。扩展的ASCII码是0~255(8位)。如果数据使用简单的文本(标准 ASCII码),那么每个数据包使用7位数据。每个包是指一个字节,包括开始/停止位,数据位和奇偶校验位。由于实际数据位取决于通信协议的选取,术语“包”指任何通信的情况。

停止位:

用于表示单个包的最后一位。典型的值为1,1.5和2位。由于数据是在传输线上定时的,并且每一个设备有其自己的时钟,很可能在通信中两台设备间出现了小小的不同步。因此停止位不仅仅是表示传输的结束,并且提供计算机校正时钟同步的机会。适用于停止位的位数越多,不同时钟同步的容忍程度越大,但是数据传输率同时也越慢。

奇偶校验位:

在串口通信中一种简单的检错方式。有四种检错方式:偶、奇、高和低。当然没有校验位也是可以的。对于偶和奇校验的情况,串口会设置校验位(数据位后面的一位),用一个值确保传输的数据有偶个或者奇个逻辑高位。例如,如果数据是011,那么对于偶校验,校验位为0,保证逻辑高的位数是偶数个。如果是奇校验,校验位为1,这样就有3个逻辑高位。高位和低位不真正的检查数据,简单置位逻辑高或者逻辑低校验。这样使得接收设备能够知道一个位的状态,有机会判断是否有噪声干扰了通信或者是否传输和接收数据是否不同步。

Qt串口通信模块QtSerialPort简介

QtSerialPort模块是QT5中附加模块的一个模块,为硬件和虚拟的串口提供统一的接口。

串口由于其简单和可靠,目前在像嵌入式系统、机器人等工业中依旧用得很多。使用QtSerialPort模块,开发者可以大大缩短开发串口相关的应用程的周期。

Qt SerialPort提供了基本的功能,包括配置、I/O操作、获取和设置RS-232引脚的信号。

Qt SerialPort模块暂不支持以下特性:

A、终端的特性,例如回显,控制CR/LF等等

B、文本模式

C、读或写操作的超时和延时配置

D、当RS-232引脚信号变化通知

要在应用程序中使用QtSerialPort,需要包括如下的声明:

#include <QtSerialPort/QtSerialPort>

要链接QtSerialPort模块,需要在.pro文件中添加如下内容:

QT += serialport

QSerialPort提供了访问串口的接口函数。使用辅助类QSerialPortInfo可以获取可用的串口信息。将QSerialPortInfo辅助类对象做为参数,使用setPort()或setPortName()函数可以设置要访问的串口设备。

设置好端口后,可以使用open()函数以只读、只写或读写的模式打开使用。

注意,串口使用独占方式打开。使用close()函数关闭串口并且取消IO操作。

串口成功打开后,QSerialPort会尝试确定串口的当前配置并初始化。可以使用setBaudRate()、setDataBits()、setParity()、setStopBits()和setFlowControl()函数重新配置端口设置。

有一对名为QSerialPort::dataTerminalReady、QSerialPort::requestToSend的属性

QSerialPort提供了中止正在调用线程直到信号触发的一系列函数。这些函数用于阻塞串口。

waitForReadyRead():阻塞调用,直到有新的数据可读

waitForBytesWritten():阻塞调用,直到数据以及写入串口

阻塞串口编程与非阻塞串口编程完全不同。阻塞串口不会要求时间循环并且通常会简化代码。然而,在GUI程序中,为了避免冻结用户界面,阻塞串口编程只能用于非GUI线程。

QSerialPort也能使用QTextStream和QDataStream的流操作符。在试图使用流操作符>>读时,需要确保有足够可用的数据。

Qt串口通信模块QSerialPort类成员函数

//构造函数
QSerialPort::QSerialPort(QObject *parent = Q_NULLPTR)
QSerialPort::QSerialPort(const QString &name, QObject *parent = Q_NULLPTR)
QSerialPort::QSerialPort(const QSerialPortInfo &serialPortInfo, QObject *parent = Q_NULLPTR)
//如果当前没有数据可读,返回true
[virtual] bool QSerialPort::atEnd() const
//波特率改变后,信号触发
[signal] void QSerialPort::baudRateChanged(qint32 baudRate, QSerialPort::Directions directions)
//返回可读数据的字节数
[virtual] qint64 QSerialPort::bytesAvailable() const
//返回可写数据的字节数
[virtual] qint64 QSerialPort::bytesToWrite() const
//关闭串口
[virtual] void QSerialPort::close()
//设置串口端口信息为serialPortInfo
void QSerialPort::setPort(const QSerialPortInfo &serialPortInfo)
//设置串口名为name
void QSerialPort::setPortName(const QString &name)

Qt串口通信模块QSerialPort简单串口实例

main.cpp代码

#include "mainwindow.h"
#include <QApplication>
int main(int argc, char *argv[])
{
 QApplication a(argc, argv);
 MainWindow w;
 w.show();
 return a.exec();
}

mainwindows.h代码参考如下:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include <QtSerialPort/QSerialPort>
#include <QtSerialPort/QSerialPortInfo>
#include <QList>
#include <QDebug>
namespace Ui {
class MainWindow;
}
class MainWindow : public QMainWindow
{
 Q_OBJECT
public:
 explicit MainWindow(QWidget *parent = 0);
 ~MainWindow();
private slots:
 void on_btn_openConsole_clicked();
 void on_btn_send_clicked();
 void on_btn_clearRecv_clicked();
 void on_btn_clearSend_clicked();
 void readData();
private:
 Ui::MainWindow *ui;
 QSerialPort *serial;
};
#endif // MAINWINDOW_H

mainwindows.cpp代码

#include "mainwindow.h"
#include "ui_mainwindow.h"
static const char blankString[] = QT_TRANSLATE_NOOP("SettingsDialog", "N/A");
MainWindow::MainWindow(QWidget *parent) :
 QMainWindow(parent),
 ui(new Ui::MainWindow)
{
 ui->setupUi(this);
 serial = new QSerialPort;
 QString description;
 QString manufacturer;
 QString serialNumber;
 //获取可以用的串口
 QList<QSerialPortInfo> serialPortInfos = QSerialPortInfo::availablePorts();
 //输出当前系统可以使用的串口个数
 qDebug() << "Total numbers of ports: " << serialPortInfos.count();
 //将所有可以使用的串口设备添加到ComboBox中
 for (const QSerialPortInfo &serialPortInfo : serialPortInfos)
 {
  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->comboBox_serialPort->addItem(list.first(), list);
 }
 ui->comboBox_serialPort->addItem(tr("custom"));
 //设置波特率
 ui->comboBox_baudRate->addItem(QStringLiteral("9600"), QSerialPort::Baud9600);
 ui->comboBox_baudRate->addItem(QStringLiteral("19200"), QSerialPort::Baud19200);
 ui->comboBox_baudRate->addItem(QStringLiteral("38400"), QSerialPort::Baud38400);
 ui->comboBox_baudRate->addItem(QStringLiteral("115200"), QSerialPort::Baud115200);
 ui->comboBox_baudRate->addItem(tr("Custom"));
 //设置数据位
 ui->comboBox_dataBits->addItem(QStringLiteral("5"), QSerialPort::Data5);
 ui->comboBox_dataBits->addItem(QStringLiteral("6"), QSerialPort::Data6);
 ui->comboBox_dataBits->addItem(QStringLiteral("7"), QSerialPort::Data7);
 ui->comboBox_dataBits->addItem(QStringLiteral("8"), QSerialPort::Data8);
 ui->comboBox_dataBits->setCurrentIndex(3);
 //设置奇偶校验位
 ui->comboBox_parity->addItem(tr("None"), QSerialPort::NoParity);
 ui->comboBox_parity->addItem(tr("Even"), QSerialPort::EvenParity);
 ui->comboBox_parity->addItem(tr("Odd"), QSerialPort::OddParity);
 ui->comboBox_parity->addItem(tr("Mark"), QSerialPort::MarkParity);
 ui->comboBox_parity->addItem(tr("Space"), QSerialPort::SpaceParity);
 //设置停止位
 ui->comboBox_stopBit->addItem(QStringLiteral("1"), QSerialPort::OneStop);
 ui->comboBox_stopBit->addItem(QStringLiteral("2"), QSerialPort::TwoStop);
 //添加流控
 ui->comboBox_flowBit->addItem(tr("None"), QSerialPort::NoFlowControl);
 ui->comboBox_flowBit->addItem(tr("RTS/CTS"), QSerialPort::HardwareControl);
 ui->comboBox_flowBit->addItem(tr("XON/XOFF"), QSerialPort::SoftwareControl);
 //禁用发送按钮
 ui->btn_send->setEnabled(false);
}
MainWindow::~MainWindow()
{
 //delete serial;
 delete ui;
}
//打开串口按钮槽函数
void MainWindow::on_btn_openConsole_clicked()
{
 qDebug() << ui->btn_openConsole->text();
 if (ui->btn_openConsole->text() == tr("打开串口"))
 {
  //设置串口名字
  serial->setPortName(ui->comboBox_serialPort->currentText());
  //设置波特率
  serial->setBaudRate(ui->comboBox_baudRate->currentText().toInt());
  //设置数据位
  serial->setDataBits(QSerialPort::Data8);
  //设置奇偶校验位
  serial->setParity(QSerialPort::NoParity);
  //设置停止位
  serial->setStopBits(QSerialPort::OneStop);
  //设置流控
  serial->setFlowControl(QSerialPort::NoFlowControl);
  //打开串口
  if (serial->open(QIODevice::ReadWrite))
  {
   ui->comboBox_baudRate->setEnabled(false);
   ui->comboBox_dataBits->setEnabled(false);
   ui->comboBox_flowBit->setEnabled(false);
   ui->comboBox_parity->setEnabled(false);
   ui->comboBox_serialPort->setEnabled(false);
   ui->comboBox_stopBit->setEnabled(false);
   ui->btn_send->setEnabled(true);
   ui->btn_openConsole->setText(tr("关闭串口"));
   //信号与槽函数关联
   connect(serial, &QSerialPort::readyRead, this, &MainWindow::readData);
  }
 }
 else
 {
  //关闭串口
  //serial->clear();
  serial->close();
  //serial->deleteLater();
  //恢复设置功能
  ui->comboBox_baudRate->setEnabled(true);
  ui->comboBox_dataBits->setEnabled(true);
  ui->comboBox_flowBit->setEnabled(true);
  ui->comboBox_parity->setEnabled(true);
  ui->comboBox_serialPort->setEnabled(true);
  ui->comboBox_stopBit->setEnabled(true);
  ui->btn_openConsole->setText(tr("打开串口"));
  ui->btn_send->setEnabled(false);
 }
}
//发送数据槽函数
void MainWindow::on_btn_send_clicked()
{
 serial->write(ui->textEdit_send->toPlainText().toLatin1());
}
//清空接收数据槽函数
void MainWindow::on_btn_clearRecv_clicked()
{
 ui->textEdit_recv->clear();
}
//清空发送区槽函数
void MainWindow::on_btn_clearSend_clicked()
{
 ui->textEdit_send->clear();
}
void MainWindow::readData()
{
 QByteArray buf;
 qDebug() << "readData: " << endl;
 buf = serial->readAll();
 if (!buf.isEmpty())
 {
  QString str = ui->textEdit_recv->toPlainText();
  str += tr(buf);
  ui->textEdit_recv->clear();
  ui->textEdit_recv->append(str);
 }
}

图形界面设计

图形界面设计如图所示:

图形界面相关属性设置:

测试结果

前提条件是需要串口硬件的支持

本文主要介绍了Qt串口通信模块QSerialPort详细使用方法与实例更多关于Qt串口通信的知识技巧请查看下面的相关链接

(0)

相关推荐

  • Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例

    首先要了解QChartView.QChart.QLineSeries.QValueAxis的实体之间的关系,例如一个QChartView中可以包含几个QValueAxis?这些可参考 Qt图形图像开发曲线图表模块QChart库基本用法.各个类之间的关系说明 每个chart可以包含多个QLineSeries数据系列,每个QLineSeries数据系列又包含了2个QValueAxis数值轴或QDateTimeAxis时间轴. 那么这个chart中的多个数据系列,一起显示在同一个chart中,会是什么

  • Qt图形图像开发之曲线图表库QtChart编译安装详细方法与使用实例

    Qt曲线图表库QtChart简介 Qt的线性绘图控件有大名鼎鼎的Qwt,ChartDirector,小巧玲珑的QCustomPlot,当然还有自家的QtChart.长久以来QtChart在Qt家族里一直是收费的模块,只有商业版才可以使用,但Qt5.7之后将开放其权限,可参见:Qt 5.7 亮瞎眼的更新.下面将介绍如何安装QtChart并进行简单的绘图. Qt曲线图表库QtChart下载 这里用git下载QtChart,参考Qt Charts 5.7.0 安装教程 Git地址:https://gi

  • Qt图形图像开发之高性能曲线图模块QCustomplot库详细使用方法与实例(支持动、静曲线图)

    Qt曲线图模块QCustomPlot库介绍 QCustomPlot是一个小型的Qt画图标类,支持绘制静态曲线.动态曲线.多重坐标曲线,柱状图,蜡烛图等 前段时间用QChart模块画图,一条曲线上面放8000条数据就会卡的不行必须要换个其他的控件,后来找到了曲线图模块QCustomplot库 这个库性能非常好,画曲线图折线图柱状图动态静态,放大缩小,都很好用,10w条数据量无压力秒画出来一点也不卡 下载地址 https://www.qcustomplot.com/index.php/downloa

  • Qt串口通信开发之QSerialPort模块详细使用方法与实例

    Qt串口通信基础及名词说明 串口通信(Serial Communications)的概念非常简单,串口按位(bit)发送和接收字节.尽管比按字节(byte)的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据.它很简单并且能够实现远距离通信.比如IEEE488定义并行通行状态时,规定设备线总长不得超过20米,并且任意两个设备间的长度不得超过2米: 而对于串口而言,长度可达1200米.典型地,串口用于ASCII码字符的传输.通信使用3根线完成,分别是地线.发送.接收.由于串口通信

  • Qt串口通信开发之Qt串口通信模块QSerialPort开发完整实例(串口助手开发)

    之前自己写了用于上位机做基本收发的界面,独立出来相当于一个串口助手,先贴图: 功能作为串口助手来说还算完善,五个发送槽,一个接收槽,可以检测可用串口并加上相关标志,串口设置,记数功能,还有菜单栏上的文件操作和一些选择功能. 下面说一说这个项目: 做这个串口助手分为两步,第一步是设计界面,第二部是功能的代码实现. 一.界面设计 界面设计用Qt Designer,当然用Qt Creator的界面编辑器也可以,只不过感觉Qt Designer更好用一点,因为可以随时运行查看你的界面效果而不用编译整个项

  • Qt图形图像开发之曲线图模块QCustomplot库生成静态、动态曲线详细教程图解

    Qt曲线图模块QCustomPlot库介绍 QCustomPlot是一个小型的Qt画图标类,支持绘制静态曲线.动态曲线.多重坐标曲线,柱状图,蜡烛图等.只需要在项目中加入头文件qcustomplot.h和qcustomplot.cpp文件,然后使一个widget提升为QCustomPlot类,即可使用. QCustomPlot官网:   http://www.qcustomplot.com/ QCustomPlot下载地址:   http://www.qcustomplot.com/index.

  • Qt图形图像开发之QT滚动区控件(滚动条)QScrollArea的详细方法用法图解与实例

    QT滚动区控件(滚动条控件)QScrollArea简介 滚动区域控件QScrollArea用于显示一个画面中的子部件的内容.如果部件超过画面的大小,视图可以提供滚动条,这样就可以看到部件的整个区域. QScrollArea属于控件容器类,可以直接在ui中拖出来. 对于QScrollArea,最难搞懂的就是:如何控制它,才能让它在我们想要出现滚动条的时候出现滚动条. 我们拖入一个QScrollArea,再向他里面拖入4个button,观察信息如下: 可以发现,4个button并不是直接位于QScr

  • Qt图形图像开发曲线图表模块QChart库缩放/平移详细方法与实例

    1.使用QChartView来缩放 (1)用鼠标框选一个矩形,把图放大到这个矩形 QChartView::setRubberBand(QChartView::RectangleRubberBand);//XY方向同时放大到鼠标画出的矩形大小(也可以设置为只放大X轴或Y轴) (2)setRubberBand函数同时也能使鼠标右键,具备缩小图的功能. 2.使用Qchart来平移和缩放 QChart::scroll(-10, 5);//整体平移(-10, 5),两个参数分别为Δx和Δy QChart:

  • Qt串口通信开发之QSerialPort模块简单使用方法与实例

    我这里主要是对串口类的简单使用,实现的功能是以读写方式打开串口,点击发送数据按钮将发送区的数据发送到缓冲区,然后在接收区显示出来,界面如下:(源码可以在这里下载) 这里使用了QSerialPort模块提供的两个类:QSerialPort类和QSerialPortInfo类,QSerialPort类提供了对串口的操作,QSerialPortInfo类提供了对串口信息的获取.下面是主要代码,包含了对串口类的简单使用. 首先,一定要在.pro文件中添加:QT += serialport 串口初始化如下

  • Qt串口通信开发之QSerialPort模块Qt串口通信接收数据不完整的解决方法

    在使用串口接收数据时,当数据量大的时候会出现数据接收不完整的情况. 因为串口数据获取函数readAll()由readyRead()信号触发,但readyRead()信号在串口读到起始标志时立即发送,并不保证一定是当前所发数据的起始部分. 因此串口通信双方在通信前应制定好通信协议,规定好数据的起始和结束标志,串口当读到完整的起始和结束标志之后,才认定读完一条完整的数据. 本例中用串口定时发送当前时间,用"#"表示数据的结尾,定时时间为0毫秒,即能发多快就发多快. 发送 void Widg

  • Qt图形图像开发之曲线图表模块QChart库读取/设置X轴的显示区间

    设置初始的显示的区间,常用的有两种方法 (1)自动 lineseries = new QLineSeries();//声明折线数据集 lineseries->append(4, 10); //填充数据集 *lineseries << QPointF(13, 5) << QPointF(17, 6); lineChart = new QChart();//创建图表 lineChart->addSeries(lineseries); // 将 数据集 添加至图表中 line

  • Qt图形图像开发曲线图表模块QChart库基本用法、各个类之间的关系说明

    如何编译安装QChart请查看下面文章 Qt图形图像开发之曲线图表库QtChart编译安装详细方法与使用实例 使用Qt曲线图表模块Chart库首先要注意3点: (1)在.pro文件中添加:QT += charts. (2)用到QChart的文件中添加:QT_CHARTS_USE_NAMESPACE,或者:using namespace QtCharts; 在ui界面中拖入一个graphicsView控件,然后右击提升为QChartView类,写提升为的类:QtCharts::QChartView

  • Qt图形图像开发之曲线图表模块QChart库坐标轴和数据不对应、密集的散点图无法显示问题解决方法

    QChart坐标轴和数据不对应问题描述: lineseries->append(4, 10); lineseries->append(5, 8); Chart->addSeries(lineseries); // 将 series 添加至图表中 axisX->setRange(0, 15);//设置X的显示范围 axisY->setRange(-20, 20); lineChart->setAxisX(axisX);//把轴添加到图表中 lineChart->se

随机推荐