超声波相关——上位机_何辞为1的博客-程序员宅基地

QT5串口编程——编写简单的上位机

https://blog.csdn.net/u014695839/article/details/50611549
首先,QT5是自带QSerialPort这个类的,使用时需要在pro文件里面添加一行:

QT       += serialport

然后直接引用头文件就可以了。

#include <QSerialPort>        //提供访问串口的功能 
#include <QSerialPortInfo>    //提供系统中存在的串口的信息

在QT5中,串口通信是借助一个QSerialPort的对象来实现的,在设置QSerialPort对象对串口的名称、波特率、数据位、校验位、停止位等参数后,方能对串口进行读写。下面,我总结了一下借助QSerailPort对串口进行设置、打开、读、写和关闭的过程。

一、设置和打开串口

//创建串口对象
QSerialPort serial;
//设置串口名
serial.setPortName("COM3");
//设置波特率
serial.setBaudRate(QSerialPort::Baud9600);
//设置数据位数
serial.setDataBits(QSerialPort::Data8);
//设置奇偶校验
serial.setParity(QSerialPort::NoParity); 
//设置停止位
serial.setStopBits(QSerialPort::OneStop);
//设置流控制
serial.setFlowControl(QSerialPort::NoFlowControl);
//打开串口
serial.open(QIODevice::ReadWrite);

以上代码是QSerialPort对象的设置示例,作用是:

  • 设置串口名为 COM3

  • 设置波特率为9600

  • 设置数据位为8位

  • 设置没有奇偶校验位

  • 设置停止位为1位

  • 设置没有流控制

  • 以可读写的方式打开串口

设置完这些就能进行读写操作了。如果遇到不理解的地方,可以选择QT的类或函数,然后按F1查阅手册。举个例子,如果我们想查看QSerialPort的其它的属性,可以选择QSerialPort的类名或成员函数,然后按F1。

二、读取数据

//从接收缓冲区中读取数据
QByteArray buffer = serial.readAll();

串口在收到数据后,会将数据存入接收缓冲区。此时,我们可以通过readAll()函数将接收缓冲区的数据读出来。当串口的接收缓冲区有数据时,QSerilaPort对象会发出一个readyRead()的信号。因此,我们可以编写一个槽函数来读数据,例如:

//连接信号和槽
QObject::connect(&serial, &QSerialPort::readyRead, this, &MainWindow::serialPort_readyRead);
 
……
 
//编写的槽函数
void MainWindow::serialPort_readyRead()
{
    //从接收缓冲区中读取数据
    QByteArray buffer = serial.readAll();
    
    //处理数据
    //……
}

三、发送数据

serial->write(data);

使用write函数便可以把字节数组中的字节发送出去。

四、关闭串口

serial->close();

串口不用时,可通过close()函数将其关闭。

接下来是一个实例

1、创建一个新的Widgets Appliaction工程

2、使用QtCreator的ui文件来设计上位机的界面,设计如下:
在这里插入图片描述

3、mainwindow.h文件内容如下:

//mainwindow.h
 
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
 
#include <QMainWindow>
#include <QSerialPort>
#include <QSerialPortInfo>
 
namespace Ui {
class MainWindow;
}
 
class MainWindow : public QMainWindow
{
    Q_OBJECT
 
public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
 
private slots:
    void serialPort_readyRead();
 
    void on_searchButton_clicked();
 
    void on_openButton_clicked();
 
    void on_sendButton_clicked();
 
    void on_clearButton_clicked();
 
private:
    Ui::MainWindow *ui;
    QSerialPort serial;
};

 
#endif // MAINWINDOW_H

4、mainwindow.cpp文件内容如下:

//mainwindow.cpp
 
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QMessageBox>
 
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
 
    //连接信号和槽
    QObject::connect(&serial, &QSerialPort::readyRead, this, &MainWindow::serialPort_readyRead);
 
    //发送按键失能
    ui->sendButton->setEnabled(false);
    //波特率默认选择下拉第三项:9600
    ui->baudrateBox->setCurrentIndex(3);
}
 
MainWindow::~MainWindow()
{
    delete ui;
}
 
void MainWindow::serialPort_readyRead()
{
    //从接收缓冲区中读取数据
    QByteArray buffer = serial.readAll();
    //从界面中读取以前收到的数据
    QString recv = ui->recvTextEdit->toPlainText();
    recv += QString(buffer);
    //清空以前的显示
    ui->recvTextEdit->clear();
    //重新显示
    ui->recvTextEdit->append(recv);
}
 
 
void MainWindow::on_searchButton_clicked()
{
    ui->portNameBox->clear();
    //通过QSerialPortInfo查找可用串口
    foreach(const QSerialPortInfo &info, QSerialPortInfo::availablePorts())
    {
        ui->portNameBox->addItem(info.portName());
    }
}
 
void MainWindow::on_openButton_clicked()
{
    if(ui->openButton->text()==QString("打开串口"))
    {
        //设置串口名
        serial.setPortName(ui->portNameBox->currentText());
        //设置波特率
        serial.setBaudRate(ui->baudrateBox->currentText().toInt());
        //设置数据位数
        switch(ui->dataBitsBox->currentIndex())
        {
        case 8: serial.setDataBits(QSerialPort::Data8); break;
        default: break;
        }
        //设置奇偶校验
        switch(ui->ParityBox->currentIndex())
        {
        case 0: serial.setParity(QSerialPort::NoParity); break;
        default: break;
        }
        //设置停止位
        switch(ui->stopBitsBox->currentIndex())
        {
        case 1: serial.setStopBits(QSerialPort::OneStop); break;
        case 2: serial.setStopBits(QSerialPort::TwoStop); break;
        default: break;
        }
        //设置流控制
        serial.setFlowControl(QSerialPort::NoFlowControl);
 
        //打开串口
        if(!serial.open(QIODevice::ReadWrite))
        {
            QMessageBox::about(NULL, "提示", "无法打开串口!");
            return;
        }
 
        //下拉菜单控件失能
        ui->portNam

eBox->setEnabled(false);
        ui->baudrateBox->setEnabled(false);
        ui->dataBitsBox->setEnabled(false);
        ui->ParityBox->setEnabled(false);
        ui->stopBitsBox->setEnabled(false);


    ui->openButton->setText(QString("关闭串口"));
    //发送按键使能
    ui->sendButton->setEnabled(true);
}
else
{
    //关闭串口
    serial.close();

    //下拉菜单控件使能
    ui->portNameBox->setEnabled(true);
    ui->baudrateBox->setEnabled(true);
    ui->dataBitsBox->setEnabled(true);
    ui->ParityBox->setEnabled(true);
    ui->stopBitsBox->setEnabled(true);

    ui->openButton->setText(QString("打开串口"));
    //发送按键失能
    ui->sendButton->setEnabled(false);
}
}
 
void MainWindow::on_sendButton_clicked()
{
    //获取界面上的数据并转换成utf8格式的字节流
    QByteArray data = ui->sendTextEdit->toPlainText().toUtf8();
    serial.write(data);
}
 
void MainWindow::on_clearButton_clicked()
{
    ui->recvTextEdit->clear();
}

5、main.cpp文件内容如下:

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

4、测试(将USB转TTL模块发送和接收引脚短接,自己发自己收)

在这里插入图片描述


做一个最简单的上位机

https://blog.csdn.net/qq_32840087/article/details/79375477
准备工作

我安装的是QT5.8,QTcreater 4.1.0。由于在官网下载比较麻烦,所以可以再这个镜像网站上下到适合自己版本的QT。

http://mirror.bit.edu.cn/qtproject/archive/qt/5.1/5.1.0/。

可能用到的软件1.串口调试助手 2虚拟串口。

然后先制作一个最简单可以查看接收数据的上位机(上面的教程)
https://blog.csdn.net/u014695839/article/details/50611549
关于Qt5 QtSerialPort串口通信:
https://blog.csdn.net/lovebird_27/article/details/49515881
Qt5实现串口通信:
https://blog.csdn.net/wamani/article/details/52849043

串口通信的概念非常简单,串口按位(bit)发送和接收字节。尽管比按字节(byte)的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据。

在低版本QT中是没有QSerialPort的串口类的,制作上位机会复杂许多。QT5是自带QSerialPort这个类的,但是在使用时,我们需要在.Pro文件中添加一行才可以使用。在之后Debug过程中如果会再报错的话,再添加到.pro文件中

QT+= core gui//使用GUI界面
QT+= serialport//使用自带串口类

打开界面文件下的.ui文件可以对我们上位机的界面进行编辑

从左侧工具栏中选择要用的工具拖到

添加textEdit 作为数据接收显示框,combobox作为串口选择框 pushubutton作为打开串口按钮,这样最基本的界面就完成了!

在这里插入图片描述

双击combox对串口进行编辑,右键点击 open port 按钮 转到槽,可以对单击按钮后的事件进行编辑。

接下来就是函数部分

在QT中我们用到了哪个函数需要在.h文件开始处声明,自定义函数以及自定义变量 都需要在头文件.h文件中事先声明,在这里声明的变量为全局变量。

首先在文件开始处添加

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

QSerialPort:提供访问串口的功能

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

1.设置串口

对串口设置如下

void weite::on_openportbutton_clicked()
{   if(ui->portnamebox->isEnabled())
    {
        ui->openportbutton->setText("ClosePort");		//按下“OpenPort”后,按键显示为“ClosePort”
        ui->portnamebox->setDisabled(true);	//按下“OpenPort”后,禁止再修改COM口
        serial1.setPortName(ui->portnamebox->currentText());	//设置COM口
        serial1.setBaudRate(QSerialPort::Baud115200,QSerialPort::AllDirections);//设置波特率和读写方向
        serial1.open(QIODevice::ReadWrite);
      //  connect(&serial1,SIGNAL(readyRead()),this,SLOT(read_com()));	//把串口的readyRead()信号绑定到read_Com()这个槽函数上
 
    }
    else
    {
        ui->openportbutton->setText("OpenPort");		//按下“ClosePort”后,按键显示为“OpenPort”
        ui->portnamebox->setEnabled(true);		//按下“ClosePort”后,COM口可被修改
        serial1.close();					//关串口
    }
}

2.接收数据

当串口收到数据并且接收完毕后,会发出一个readyRead()的信号,因此只需要编写一个槽函数read_com(),设置信号槽,并在槽函数中使用readAll()把收到的数据读到requestData中。

3.数据显示

这里我们用了自带的append 函数来显示收到的数据 ,使用toHex()函数来将收到的8位ascii码 转换为16进制字符型显示

(在使用QT过程中在使用某个函数前,可以上网搜索其功能,QT中也自带帮助功能,按下F1可以查看某个函数功能及示例)

2和3部分代码如下

void weite::read_com()
{
    QByteArray requestData;
    requestData= serial1.readAll();
// qDebug()<<"---hsy--test--showtable---1";
  QString buf ;
   // requestData.clear();
   ui->textEdit->append(requestData.toHex());
}

到这里我们最简单的上位机制作便完成了,可以连接上不需要启动信息的(串口设备)传感器来测试我们上位机效果。


IMU传感器的上位机

和最简单的上位机比起来,我们需要增加一个数据解析部分,也就是将接受到的传感器信息进行译码,得到我们可以直接读取的信息。我们可以从传感器的说明书上得到数据的通讯协议 比如这个

在这里插入图片描述

那么从这里我们可以知道 以A5 5A为头 AA为尾的这样一组数据就是我们要进行译码的数据。

我们利用readall()函数读到的的内容是一存到了requestData中,数据类型为Qbytearry(数组类型Qbytearry为QT中独特的数据类型,既可以存储字符串,又可以存储数)我们可以转化为整型(int类型)再进行进一步解算。(对于计算机来说0x01和1没有区别)

这里先将requestData存入缓冲区然后进行转存到str(string字符串类型存入其中数据以字符串形式存入)中,由于每一段的数据帧的长度为20个字节所以识别头0xaa到尾0xa2的长度应该为20字节然后 这二十个字节的数据分别存到了record[1]到record[20]中,令尾部0xaa为k那么距离他21个字节处便是0xa2,record1[k-19]和record1[k-19]便分别为加速度数据的高8位和低8位,高8位乘上256(2的8次方)加上低8位=t1.t1便是带符号的最终数据,最后将t1与0x8000做与运算判断最高位符号位,并做处理。

最终显示在面板上的数据 加速x.对其他数据依次做这样处理,依次显示在面板上.效果图如下。

在这里插入图片描述

ACX.sprintf("%f",t1*(9.8)/4089);


 if(!requestData1.isEmpty())
 {  long len,k;
     QBuffer buffer(&requestData1);
     //只写模式打开缓冲区
      buffer.open(QIODevice::ReadWrite);
 buffer.write(requestData1);
  if(!requestData1.isEmpty())
  {
 //qDebug()<<"1";
      static quint8 record[2048];
      static quint8 record1[2048];
      const char*str =buffer.data();
      const char*str1 =buffer.data();
      len =buffer.size();
     for(k=0;k<len;k++)
     {
        // qDebug()<<"2";
        record1[k]=(quint8)str1[k];
        if (record1[k]== 0xAA)
        { if(record1[k-20]== 0xA2)
            {  //qDebug()<<"2";
                int t1=record1[k-19]*256+record1[k-18];
                if(t1&0x8000)
                   {
                    t1 = 0-(t1&0x7fff);
                   } else
                t1 = (t1&0x7fff);
                ACX.sprintf("%f",t1*(9.8)/4089);
                ui->AX->setText(ACX);

激光雷达部分

阅读说明书可以看到雷达基本通讯模式“与 RPLIDAR 进行的通讯采用非文本形式的二进制数据报文进行,且每个数据报文均具有统一的报头数据格式。外部系统在发送开始扫描的请求后,RPLIDAR 将开始连续的扫描测距。在每次测距操作完成后,对应的测距采样点的信息(距离、角度等)将通过一个独立应答包的形式发送至外部系统。在这个模式下,外部系统只需要发送单次的请求,并开始连续接受来自 RPLIDAR 的多个应答数据文报。”

请求报文格式
在这里插入图片描述

常用请求报文

在这里插入图片描述

开始扫描采样(SCAN)命令请求与回应数据格式:

在这里插入图片描述

PLIDAR 工作在空闲状态时,在外部系统发送了该请求后,将开始进入测距采样。每个测距采样点将使用数据应答报文发送至外部系统。如果 RPLIDAR先前已经工作在测距采样状态,则 RPLIDAR 首先将停止正在进行的测距采样功能,并重新开始新一轮的测距采样操作。

![在这里插入图片描述](https://img-blog.csdnimg.cn/20190725200203595.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80NTQzODY1Mw==,size_16,color_FFFFFF,t_70)

在收到起始应答回复之后雷达开始不断返回采集的数据。每五个字节为一组,第一个字节第一位是标志位,第二位是取标志位的反,他们两个在任何时刻一个为0,另一个必为1,第二个字节第一位是校验位,通过这三个字节我们可以验证编写的是否有错误。第一字节的第三位到第8位是信号质量。第二个字节的第二位到第8位为角度angel的低7位,第3个字节为第8位到第14位。

第四个字节和第五个字节分别为距离distance的低8位与高8位。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
数据的存储

按下界面上的开始采集按钮,同时开始IMU与激光雷达数据采集,IMU数据的采集,采集的面板数据,采用定时器Qtimer定时,每20ms向IMU数据中写入换行符 进行换行。每行

激光雷达则是将换行符写入了存储数据的循环内,每行为512个采样点,约为0.25s。

激光雷达数据

void Widget::on_kscj_clicked()
{//在c盘下创建以开始采集时间命名的激光雷达数据文档
    serial.open(QIODevice::ReadWrite);
    QDateTime time = QDateTime::currentDateTime();
    QString date = time.toString("yyyy-MM-dd hh-mm-ss "); //设置显示格式
    fileName = "C:/" + date + ".txt";
    file.setFileName(fileName);
    if(!file.open(QIODevice::ReadWrite | QIODevice::Text))
              serial.write(StringToHex("A5 25"));
              serial.write(StringToHex("A5 20"));
              ui->stateLab->setText("当前状态 : 普通扫描指令已经发送");
//在d盘下创建以开始采集时间命名的IMU数据文档
    fileName1 = "D:/" + date + ".txt";//
file1.setFileName(fileName1);
    }

杜洋的简单上位机制作
https://www.bilibili.com/video/av38980254?from=search&seid=7839098653315187718

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_45438653/article/details/97295270

智能推荐

[论文阅读] | MobileNets & depthwise separable convolutions_风中凌乱的小精灵的博客-程序员宅基地

写在前面:遇到不会的点,要督促自己及时查漏补缺。上周六涉及到深度可分离卷积,涉及到了我的知识盲区,于是昨天找了 MobileNets 相关的笔记和论文,重点在于理解可分离卷积的设计思想。本篇笔记更新记录:2021.12.15 整理可分离卷积、以及在其基础上建立的MobileNets的基本概念。目录1 整体思想2 巨人的肩膀——在轻量级模型领域的已有工作3 核心——深度可分离卷积3.1 标准的卷积运算3.2 深度可分离卷积4 总结1 整体思想出发点:在CV领域,一个研究趋势是神经网络层数越来.

mysql怎么 执行_一条查询语句在MySQL中是如何执行的?_且听且说吧的博客-程序员宅基地

前言我们在学习一种技术的时候,首先要鸟瞰其全貌,千万不要一开始就陷入到细节中去,这样有助于我们站在高维度其理解问题 —— 丁奇。学习MySQL也是一样,所以我们可以从一条查询语句的执行开始看起。select * from t where id = 1;通常情况下,我们在使用MySQL的时候,只是从客户端输入一行指令,然后获取一个返回结果。但是对于一个开发人员来说,只知道这些是远远不够的,我们还需要...

multipart文件上传大小受限_muxiaohuadu的博客-程序员宅基地

我们的项目框架是spring boot,容器是tomcat,代理是nginx文件上传时如果报类似跨域问题的错误提示时,如果小点的图片能够上传说明不是跨域问题,这时候需要思考为什么大的图片无法上传我今天就踩了这个坑,并且已经解决首先配置springboot的multipart参数在propertiesmultipart.maxFileSize=10MBmultipart.maxReque...

shell变量嵌套_半土的博客-程序员宅基地_shell变量嵌套变量

闲话少说,翻了很多百度,发现一些问题:想用二维数组,但是shell支持的不好于是只能:在变量套变量的调用中也走了一些弯路,不过还好解决了,总结如下:两个关键,第一个是eval:在for循环中,数组变量取值(数组变量中嵌套了变量)时加入了evaleval nList$j[$k]=`echo $i|cut -d, -f$k`第二个是shell不支持${{ }},所以在加入e...

自然语言交流系统 phxnet团队 创新实训 项目博客 (十三)_weixin_30475039的博客-程序员宅基地

对我们项目中的关键技术实现进行总结:一、3DMax关键技术实现1、一下的关于3DMax中对于人物的设计和操作均需要在对3DMax基础知识熟练掌握的情况下进行的。2、 骨骼架设:首先对导入到3DMax中的人物模型进行架设骨骼,首先,先加载一个人,锁定住,别让他乱动。用biped工具建立一个基本骨骼——可以从脚部位置往上拖拽鼠标来建立。在运动命令面板,点biped卷展栏的fi...

随便推点

关于UNDO表空间及undo_retention参数的一点研究_cuanshenqiao1119的博客-程序员宅基地

对于UNDO表空间大小的定义需要考虑UNDO_RETNETION参数、产生的UNDO BLOCKS/秒、UNDO BLOCK的大小。undo_retention:对于UNDO表空间的数据文件属性为autoextensible,则...

一小例子,了解 TCP 通讯流程 | Qt 示例_嵌入式小傻瓜的博客-程序员宅基地

Hi,我是你们的工具人,老吴。今天用一个小例子,陈述一下 Qt 里使用 TCP 通讯的流程。代码链接:https://doc.qt.io/qt-5/examples-network.htm...

html5 jquery设置vaild,jQuery .valid()无效? (验证插件)_半个科创史学先生的博客-程序员宅基地

我使用jQuery Validation Plugin并且我的代码没有按预期工作。我想要的是当我单击提交按钮时,程序将检查表单是否正确填写并包含一个URL然后运行一个功能。在这种情况下,它总是alert('true)。$(document).ready(function(){$('#button1').submit(function() {if($("#button1").valid()){ale...

python界面添加图片_python打开图片放到tkinter的界面里(opencv方式)_weixin_39633493的博客-程序员宅基地

上次使用的别人的代码打开图片放到tkinter的图型界面当中。虽然感觉没有很懂但是不是自己想用的模块。今天换成了opencv的方式,修改成功。代码如下:# -*- coding: utf-8 -*-"""Created on Sat Mar 7 18:03:50 [email protected]: liyan"""import tkinterfrom tkinter import *from PIL im...

多个Tomcat的情况下,不同的Tomcat总是启动某一个Tomcat的错误_栎秋的博客-程序员宅基地

问题:无论启动8081还是8082都是启动的8.5.6。解决方式可以有以下几种:找到 环境变量里的 CATALINA_HOME,将其删除,如果Path里也有相关配置的话,也删除。打开8081或8082 的startup.bat文件 找到: Guess CATALINA_HOME if not defined set "CURRENT_DIR=%cd%" if not "%CATALINA_HOME%" == "" goto gotHome set "CAT.

2018.2.12_swy_swy_swy的博客-程序员宅基地

2018.2.12这是一个内容预告本报告介绍一次有关数据分类预测以及聚类分析的实验,实验基于python sklearn库,并提供了实验者自己对于朴素贝叶斯算法以及K-Means聚类算法的实现。之后基于graphviz可视化决策树,基于t-sne和matplotlib对聚类结果进行降维与可视化。最后使用10折交叉验证对分类预测结果进行评价;使用Fowlkes-Mallows 指数(以下简称f-...