[Qt开发]一口气搞懂串口通信
好多小鳄鱼
一、关于串口通信:
Qt的确有自己的串口通信类,就是QSerialPort,但是我们在使用过程中因为要更加定制化的使用串口通信类减小开发的难度,所以我们会提供一个串口通信类,也就是这个SerialPortHelper类。
首先我们要知道什么是串口,串口通信就是机器和系统之间的一个通信协议,你可以将它理解为共享内存,可以根据需要向其中写入内容,然后在需要的时候从中读取数据。不过需要注意的是,在Qt的封装下,你不需要知道串口内的数据是否是给你的,还是你发的,因为统统都是你的。
串口通信什么?
大概了解了一下什么是串口通信,那么我们来看一下串口通信的通信手册大概是什么样。

由上我们可以看到,串口通信消息大概就是一串16进制的字符按照特定的规定,然后向串口中写入这些消息就可以了。比如前面这三个41 54 64这三个数字就是固定写死的,而36则是这个协议的通信号,是用于区分不同消息码的,比如上述这条指令的消息码是36,另外一条消息
然后后续标了颜色的内容就是实际上这条消息码中具体携带的数据,这个就不过多介绍了。
不过需要注意的是,硬件返回的数据或者向硬件发送的数据不一定都是像人一样的从左到右,有些命令特别是偏长的命令很多都是要求从右到左,也就是常说的小端序,低字节在前高字节在后,比如:

给出两端函数示例(实际上这两个函数在Qt中都有,这里只是展示一下具体是什么意思而已):
//将接收到的小端字节序数据转换为无符号整数
QString SerialPortHelper::getLittleEnd(const QByteArray& data)
{
if (data.size() > 8) return "";
qulonglong result = 0;
for (int i = 0; i < data.size(); ++i)
{
qulonglong tmp = (uchar)data[i];
result += tmp <<= i * 8;
}
return QString::number(result);
}
//大端字节序数据转换为浮点数
QString SerialPortHelper::getBigEndFlt(const QByteArray& data)
{
const int fltLen = 4;
if (data.size() != fltLen) return "null";
float result = 0;
uchar fltArr[fltLen];
for (int i = 0; i < fltLen; ++i)
{
fltArr[fltLen - i - 1] = data.at(i);
}
return QString::number(*(float*)fltArr, 'f', 9);
}
ok,接下来还有四位数字,这个要分开来说前两位和后两位。
其中前两位是CRC校验码,这个是需要对前面的数字进行一个基本的校验,具体我就不太懂了,这里提供一个函数,供参考:
void SerialPortHelper::CRC16_2(const QByteArray& ba, uchar* crcBuf)
{
int pos, i;
uchar* buf = (uchar*)ba.data();
int len = ba.size();
unsigned int crc = 0xFFFF;
for (pos = 0; pos < len; pos++)
{
crc ^= (unsigned int)buf[pos]; // XOR byte into least sig. byte of crc
for (i = 8; i != 0; i--) // Loop over each bit
{
if ((crc & 0x0001) != 0) // If the LSB is set
{
crc >>= 1; // Shift right and XOR 0xA001
crc ^= 0xA001;
}
else // Else LSB is not set
{
crc >>= 1; // Just shift right
}
}
}
//高低字节转换
crc = ((crc & 0x00ff) << 8) | ((crc & 0xff00) >> 8);
//qDebug() << QString().sprintf("CRC:%04x", crc);
crcBuf[0] = crc >> 8;
crcBuf[1] = crc;
}
最后两位就是固定的了,用两个固定搭配来分割各种字符段
串口通信使用流程
我们一般使用串口类,主要流程如下:
1.获得基本参数:
我们需要获得这个串口的一些基本参数,其中包含内容如下:
QString portName = "NULL";
int baudRate = 921600;
QSerialPort::DataBits dataBits = QSerialPort::Data8;
QSerialPort::StopBits stopBits = QSerialPort::OneStop;
QSerialPort::Parity parity = QSerialPort::NoParity;
我们需要在启动这个端口调用的时候设置好这些属性,才能获取正确的COM口消息和发送消息,给出一个启动串口的示例:
serialPort = new QSerialPort();
serialPort->setPortName(param.portName);
if (serialPort->open(QIODevice::ReadWrite))
{
serialPort->setBaudRate(param.baudRate);
serialPort->setDataBits(param.dataBits);
serialPort->setStopBits(param.stopBits);
serialPort->setParity(param.parity);
//通知串口接收到消息的信号函数
connect(serialPort, &QSerialPort::readyRead, this, &CarControlModel::dataReceive);
qDebug() << "connect:" << param.portName;
emit checkConnableRetSig(true);
setStartTime();
//emit comConnStatus(true);
return true;
See?其实很简单的的,这个对象实际上就做了一件事,通知你现在来数据了。注意,这个readyRead 并不是发给你接到的数据,而是通知你现在串口接到消息了,而且这个消息还很有可能不是一条一条的,可能是一段一段的,就是消息可能不完整,这里给出一个示例。
void CarControlModel::dataReceive()
{
if (serialPort != nullptr && serialPort->isOpen())
{
QByteArray buffer = serialPort->readAll();
analysisData(buffer);
}
}
void CarControlModel::analysisData(const QByteArray& dataArr)
{
//消息断开的情况
qDebug() << "recv data: " << dataArr.toHex(' ');
dataBuff.append(dataArr);
QList<QByteArray> dataList;
for (;;)
{
int index = dataBuff.indexOf(QByteArray(gEnd, gEndLen));
if (index == -1) break;
dataList.append(dataBuff.mid(0, index + gEndLen));
dataBuff = dataBuff.right(dataBuff.size() - index - gEndLen);
}
if (dataList.size() == 0) return;
。。。。
也就是说,接到串口指令之后可能还需要你做一个解析,把接到的消息拆分出来处理。
向串口发送指令:
发送指令的话,就比较简单了,比如:
qint64 CarControlModel::startCentralCMD()
{
QByteArray allArr, funDataArr;
funDataArr.clear();
funDataArr.append(0x32);
funDataArr.append(0x01); //0x01启动中心码盘
code(allArr, funDataArr);
return writeData(allArr);
}
//传入功能码数据区,组合成完整的指令存入数组
void CarControlModel::code(QByteArray& allArr, const QByteArray& funDataArr)
{
allArr.clear();
allArr.append(gId, gIdLen);
allArr.append(gAddress, gAddressLen);
allArr.append(funDataArr);
uchar crcBuf[gCrcLen];
CRC16_2(allArr, crcBuf);
allArr.append((char*)crcBuf, gCrcLen);
allArr.append(gEnd, gEndLen);
}
//串口发送消息出去
qint64 CarControlModel::writeData(const QByteArray& data)
{
qDebug() << "send data: " << data.toHex(' ');
qint64 ret = -1;
if (serialPort != nullptr && serialPort->isOpen())
{
ret = serialPort->write(data);
if (!serialPort->waitForBytesWritten(10000))
{
//emit sendWarningSig("发送失败:" + data.toHex(' '));
qDebug() << QString("发送失败:" + data.toHex(' '));
}
}
else
{
emit sendWarningSig("工控主串口未连接");
}
return ret;
}
[Qt开发]一口气搞懂串口通信的更多相关文章
- 【Qt开发】Qt5.7串口开发
QT5有专门的串口类: QSerialPort:提供访问串口的功能 QSerialPortInfo:提供系统中存在的串口的信息 具体使用方法: 1.在pro文件中加入: QT += seria ...
- 微信小程序开发之搞懂flex布局1——Flexbox
Flexbox ——弹性布局 Flexbox is a layout model for displaying items in a single dimension — as a row or as ...
- 微信小程序开发之搞懂flex布局2——flex container
容器的概念,是用来包含其它容器(container)和项目(item). flex container——flex容器 A flexbox layout is defined using the fl ...
- 微信小程序开发之搞懂flex布局3——Flex Item
Flex Item flex容器的子元素就是这个容器的flex item. The direct children of a Flex Container (elements with display ...
- 微信小程序开发之搞懂flex布局4——Main Axis
Main Axis——主轴 当flex-direction设置为row或row-reverse时,主轴的方向为水平方向.此时flex item为行内级元素. 当flex-direction设置为col ...
- 微信小程序开发之搞懂flex布局5——cross axis
Cross Axis——交叉轴,与Main Axis(主轴)垂直交叉. main axis is row or row-reverse the cross axis runs down the col ...
- STM32F407 串口通信实验 视频第27节 个人笔记
前言 第26节也是串口,笔记链接在此:https://www.cnblogs.com/YuQiao0303/p/10019362.html github地址:https://github.com/Yu ...
- Qt编写串口通信程序全程图文解说
(说明:我们的编程环境是windows xp下,在Qt Creator中进行,假设在Linux下或直接用源代码编写,程序稍有不同,请自己修改.) 在Qt中并没有特定的串口控制类,如今大部分人使用的是第 ...
- 转:Qt编写串口通信程序全程图文讲解
转载:http://blog.csdn.net/yafeilinux/article/details/4717706 作者:yafeilinux (说明:我们的编程环境是windows xp下,在Q ...
- Qt串口通信
1. Qt串口通信类QSerialPort 在Qt5的的更新中,新增了串口通信的相关接口类QSerialPort,这使得在开发者在使用Qt进行UI开发时,可以更加简单有效地实现串口通信的相关功能. 开 ...
随机推荐
- Windows下解决python pip命令下载慢的方法(超简单)
首先,常用的国内镜像: 1 https://pypi.tuna.tsinghua.edu.cn/simple/ 清华大学(推荐) 2 http://pypi.douban.com/simple/ 豆瓣 ...
- 验证ADG的坏块检测和自动修复
环境: Oracle 19c ADG(主库:单实例:备库:RAC) 1.主库新建测试文件 2.主库创建测试表 3.查询表对应数据文件信息 4.模拟数据文件物理坏块 5.查询对应测试表 6.进一步查询日 ...
- kali 安装beef-xss (含输入正确密码却登录不了本机无法打开beef提示找不到服务器404常见解决办法)
1:安装beef 2:修改配置文件 3:浏览器打开地址127.0.0.1:3000/ui/panel 常见问题解决本文最后 kali 打开终端 输入apt-get install beef-xss - ...
- 第一章 static、单例与继承
目录 面向对象 一.static关键字 1.static修饰成员变量 2.static修饰成员变量内存中执行原理 3.成员方法的执行原理 4.工具类 5.静态关键字注意事项 6.代码块 java静态代 ...
- Gpssworld仿真(二):并排排队系统模拟
4.3 某一个加油站能够配给三个级别的燃油:①家庭取暖用的燃油:②轻工业用的燃油:③运输用的燃油.每一级别的燃油都有一个对应的油泵.订单中燃油的数量在3000加仑和5000加仑中变化,每次增加10加仑 ...
- mysql数据库的登录脚本
######################## ku脚本: 可以使用以下ku脚本,它可以根据提供的参数登录到MySQL数据库: #!/bin/bash # Check for correct num ...
- JMeter-BeanShell预处理程序和BeanShell后置处理程序的应用
一.什么是BeanShell? BeanShell是用Java写成的,一个小型的.免费的.可以下载的.嵌入式的Java源代码解释器,JMeter性能测试工具也充分接纳了BeanShell解释器,封装成 ...
- Vulnhub Development Walkthrough
Vulnhub Development Walkthrough Recon 首先使用netdiscover进行二层Arp扫描. ┌──(kali㉿kali)-[~] └─$ sudo netdisco ...
- 百度松果菁英班--oj赛(第二次)
目录 一.小码哥剪绳子 二.咖啡品鉴师小码哥 三.均分糖果 四.持盾 五.活动安排 六.甜品供应 七.斐波那契数列的组合 八.小码哥的布阵指挥 九.活动分组 十.外卖递送 一.小码哥剪绳子 题目:马上 ...
- Linux云计算运维工程师day28shell编程基础
一. 1.全局变量.环境变量 Export OLDOBY="I am a oldboy." Echo OLDOBY OLDOBY="I am a oldboy.&quo ...