技术专栏
【海思HI3520之QT开发】-串口通讯实战(三):泰斗N303-3数据采集
环境:Ubuntu 12.04-64bit
硬件平台:Hi3520D_V100
内核版本:linux-3.0.y
Qt版本:qt4.8.6
编译器:arm-hisiv100nptl-linux-gcc
作者:MacianYuan
原文链接:https://www.ebaina.com/articles/140000004228
摘要:
1、泰斗参数配置
2、解决Qt串口通信接收数据不完整的两种方法
第一节 泰斗参数配置
一、 泰斗N303-3资料
官方用户手册:
https://www.ebaina.com/resources/240000028466
网上比较火的调试心得:
https://www.ebaina.com/resources/240000028467
二、主要字段分析
$GPGGA
例:$GPGGA,092204.999,4250.5589,S,14718.5084,E,1,04,24.4,19.7,M,,,,0000*1F
字段0:$GPGGA,语句ID,表明该语句为Global Positioning System Fix Data(GGA)GPS定位信息
字段1:UTC 时间,hhmmss.sss,时分秒格式
字段2:纬度ddmm.mmmm,度分格式(前导位数不足则补0)
字段3:纬度N(北纬)或S(南纬)
字段4:经度dddmm.mmmm,度分格式(前导位数不足则补0)
字段5:经度E(东经)或W(西经)
字段6:GPS状态,0=未定位,1=非差分定位,2=差分定位,3=无效PPS,6=正在估算
字段7:正在使用的卫星数量(00 - 12)(前导位数不足则补0)
字段8:HDOP水平精度因子(0.5 - 99.9)
字段9:海拔高度(-9999.9 - 99999.9)
字段10:地球椭球面相对大地水准面的高度
字段11:差分时间(从最近一次接收到差分信号开始的秒数,如果不是差分定位将为空)
字段12:差分站ID号0000 - 1023(前导位数不足则补0,如果不是差分定位将为空)
字段13:校验值
$GPGLL
例:$GPGLL,4250.5589,S,14718.5084,E,092204.999,A*2D
字段0:$GPGLL,语句ID,表明该语句为Geographic Position(GLL)地理定位信息
字段1:纬度ddmm.mmmm,度分格式(前导位数不足则补0)
字段2:纬度N(北纬)或S(南纬)
字段3:经度dddmm.mmmm,度分格式(前导位数不足则补0)
字段4:经度E(东经)或W(西经)
字段5:UTC时间,hhmmss.sss格式
字段6:状态,A=定位,V=未定位
字段7:校验值
$GPGSA
例:$GPGSA,A,3,01,20,19,13,,,,,,,,,40.4,24.4,32.2*0A
字段0:$GPGSA,语句ID,表明该语句为GPS DOP and Active Satellites(GSA)当前卫星信息
字段1:定位模式,A=自动手动2D/3D,M=手动2D/3D
字段2:定位类型,1=未定位,2=2D定位,3=3D定位
字段3:PRN码(伪随机噪声码),第1信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段4:PRN码(伪随机噪声码),第2信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段5:PRN码(伪随机噪声码),第3信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段6:PRN码(伪随机噪声码),第4信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段7:PRN码(伪随机噪声码),第5信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段8:PRN码(伪随机噪声码),第6信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段9:PRN码(伪随机噪声码),第7信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段10:PRN码(伪随机噪声码),第8信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段11:PRN码(伪随机噪声码),第9信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段12:PRN码(伪随机噪声码),第10信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段13:PRN码(伪随机噪声码),第11信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段14:PRN码(伪随机噪声码),第12信道正在使用的卫星PRN码编号(00)(前导位数不足则补0)
字段15:PDOP综合位置精度因子(0.5 - 99.9)
字段16:HDOP水平精度因子(0.5 - 99.9)
字段17:VDOP垂直精度因子(0.5 - 99.9)
字段18:校验值
$GPGSV
例:$GPGSV,3,1,10,20,78,331,45,01,59,235,47,22,41,069,,13,32,252,45*70
字段0:$GPGSV,语句ID,表明该语句为GPS Satellites in View(GSV)可见卫星信息
字段1:本次GSV语句的总数目(1 - 3)
字段2:本条GSV语句是本次GSV语句的第几条(1 - 3)
字段3:当前可见卫星总数(00 - 12)(前导位数不足则补0)
字段4:PRN 码(伪随机噪声码)(01 - 32)(前导位数不足则补0)
字段5:卫星仰角(00 - 90)度(前导位数不足则补0)
字段6:卫星方位角(00 - 359)度(前导位数不足则补0)
字段7:信噪比(00-99)dbHz
字段8:PRN 码(伪随机噪声码)(01 - 32)(前导位数不足则补0)
字段9:卫星仰角(00 - 90)度(前导位数不足则补0)
字段10:卫星方位角(00 - 359)度(前导位数不足则补0)
字段11:信噪比(00-99)dbHz
字段12:PRN 码(伪随机噪声码)(01 - 32)(前导位数不足则补0)
字段13:卫星仰角(00 - 90)度(前导位数不足则补0)
字段14:卫星方位角(00 - 359)度(前导位数不足则补0)
字段15:信噪比(00-99)dbHz
字段16:校验值
$GPRMC
例:$GPRMC,024813.640,A,3158.4608,N,11848.3737,E,10.05,324.27,150706,,,A*50
字段0:$GPRMC,语句ID,表明该语句为Recommended Minimum Specific GPS/TRANSIT Data(RMC)推荐最小定位信息
字段1:UTC时间,hhmmss.sss格式
字段2:状态,A=定位,V=未定位
字段3:纬度ddmm.mmmm,度分格式(前导位数不足则补0)
字段4:纬度N(北纬)或S(南纬)
字段5:经度dddmm.mmmm,度分格式(前导位数不足则补0)
字段6:经度E(东经)或W(西经)
字段7:速度,节,Knots
字段8:方位角,度
字段9:UTC日期,DDMMYY格式
字段10:磁偏角,(000 - 180)度(前导位数不足则补0)
字段11:磁偏角方向,E=东W=西
字段16:校验值
$GPVTG
例:$GPVTG,89.68,T,,M,0.00,N,0.0,K*5F
字段0:$GPVTG,语句ID,表明该语句为Track Made Good and Ground Speed(VTG)地面速度信息
字段1:运动角度,000 - 359,(前导位数不足则补0)
字段2:T=真北参照系
字段3:运动角度,000 - 359,(前导位数不足则补0)
字段4:M=磁北参照系
字段5:水平运动速度(0.00)(前导位数不足则补0)
字段6:N=节,Knots
字段7:水平运动速度(0.00)(前导位数不足则补0)
字段8:K=公里/时,km/h
字段9:校验值
三、在Qt中配置泰斗
-
需要信息的结构体
struct GPS_information{ QString GPS_Latitude;//纬度------------------------GPRMC QString GPS_Longitude;//经度-----------------------GPRMC QString GPS_status;//GPS状态 A 有效定位 V无效定位----------GPRMC QString GPS_num_satellites;//GPS可见卫星数----------------GPGGA QString GPS_speed;//GPS地面速率---------------------------GPVTG };
-
因为项目中需要GPS信息有经纬度、GPS状态、可见卫星数、地面速率所以因为项目中需要GPS信息有经纬度、GPS状态、可见卫星数、地面速率所以通过以下配置配置模块功能。
void GpsAnalysis::Gps_set()
{
WriteCom(Device_public::gps_port,"$CCINV,1000,*50\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,RMC,1,1,*05\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,VTG,1,1,*1C\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GGA,1,1,*19\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GSA,1,0,*0D\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GSV,1,0,*1A\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GLL,1,0,*1F\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,ZDA,1,0,*07\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,DTM,1,0,*05\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GNS,1,0,*02\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GBS,1,0,*0E\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GRS,1,0,*1E\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,GST,1,0,*18\r\n");
WriteCom(Device_public::gps_port,"$CCMSG,TXT,1,0,*00\r\n");
}
- 标准的串口写函数
int GpsAnalysis::WriteCom(QextSerialPort *write_com,QString str)
{
if(write_com == 0 || !write_com->isOpen()){
qDebug() << "The write_com Open Failed!";
return -1;
}
int type_num;
QByteArray array= str.toUtf8();
type_num = write_com->write(array);
if(type_num <= 0){
//QMessageBox::information(0,tr("错误"),tr("通讯中断"),QMessageBox::Ok);
write_com->close();
return -1;
}
return 0;
}
- 在构造函数中初始化串口,使用波特率9600:
GpsAnalysis::GpsAnalysis(QObject *parent) :
QObject(parent)
{
PortSettings settings = {BAUD9600, DATA_8, PAR_NONE, STOP_1, FLOW_OFF, 10};
//if (!Device_public::gps_port->isOpen()) {
Device_public::gps_port = new QextSerialPort("/dev/ttyAMA1", settings, QextSerialPort::EventDriven);
Device_public::gps_port->open(QIODevice::ReadWrite);
//}
//if (Device_public::gps_port->isOpen() && Device_public::gps_port->queryMode() == QextSerialPort::EventDriven){
//定时100ms的数据
timer = new QTimer(this);
timer->setInterval(100);
//GPS 配置
Gps_set();
//有数据读取到,进入数据分析槽函数,分析出数据
connect(Device_public::gps_port, SIGNAL(readyRead()), this,SLOT(ReadCom()));
connect(timer, SIGNAL(timeout()), this,SLOT(on_timeout()));
//}
}
第二节 解决Qt串口通信接收数据不完整的两种方法
四、第一种以定时器的方式获取整个数据包
思路:
- 通过上面构造函数 和 下面 读分析槽函数 和 定时器槽函数 进行收取完整包。
- 当有数据到达时,启动定时器定时100ms不断接收数据追加到baRcvData字节数组中
- 100ms之后在进行数据分析,数据分析通过使用QStringList把数据分解到QString中进行解析
注:定时器时间的配置:通过示波器观察我发了10字节的数据,大概在10ms,基本上可以计算为1字节1ms,所以通过这个可以大概推算整个包接收的时间。
如果移植到其他传感器采集注意两组包中间的间隔时间要大于100ms
//定时100ms,不断把数据接收追加到数组
void GpsAnalysis::ReadCom()
{
//启动定时器,接收100毫秒数据
timer->start();
baRcvData.append(Device_public::gps_port->readAll());
}
//读数据 对数据进行解析
void GpsAnalysis::on_timeout()
{
QString RMC_data;
QString GGA_data;
QString VTG_data;
timer->stop();
if(baRcvData.length()!=0) {
if(baRcvData.startsWith("$")){
RMC_data = baRcvData.mid(baRcvData.indexOf("$GNRMC,"),baRcvData.indexOf("$GNGGA,"));
GGA_data = baRcvData.mid(baRcvData.indexOf("$GNGGA,"),baRcvData.indexOf("$GNVTG,"));
VTG_data = baRcvData.mid(baRcvData.indexOf("$GNVTG,"),baRcvData.size());
}
QStringList RMC_list = RMC_data.split(",");//QString字符串分割函数
QStringList GGA_list = GGA_data.split(",");//QString字符串分割函数
QStringList VTG_list = VTG_data.split(",");//QString字符串分割函数
if((RMC_list.size()>8) && (VTG_list.size()>8)){
Device_public::gps_information.GPS_status = RMC_list.at(2);
//有效定位 定位成功
if(Device_public::gps_information.GPS_status.contains("A")){
Device_public::gps_information.GPS_Latitude = RMC_list.at(3);
Device_public::gps_information.GPS_Longitude = RMC_list.at(5);
Device_public::gps_information.GPS_num_satellites = GGA_list.at(7);
Device_public::gps_information.GPS_speed = VTG_list.at(7);
qDebug()<<Device_public::gps_information.GPS_Latitude ;
qDebug()<<Device_public::gps_information.GPS_Longitude ;
qDebug()<<Device_public::gps_information.GPS_num_satellites ;
qDebug()<<Device_public::gps_information.GPS_speed ;
}
//无效定位 正在定位
else if(Device_public::gps_information.GPS_status.contains("V")){
}
}
//获取GPS信息失败
else{
}
}
baRcvData.clear();
}
第二种,以判断包头包尾的形式,追加到字节数组中。这个方法是我之前与单片机通讯时用的的,贴出来参考。
//读数据 对数据进行解析
static QByteArray BufferData,PasteData,ReadData,AddData;
void SerialCommunication::ReadCom()
{
if (Device_public::key_port->bytesAvailable() <= 0){
qDebug() << "read com = 0";
return ;
}
//接收的数据存储到BufferData中
BufferData = Device_public::key_port->readAll();
//定义包头和包尾,我这边接收的是json包所以定义如下
const QByteArray buff_head = "$";
const QByteArray buff_end = "}}";
//数据判断
//异常类:无头且变量为空,已丢失头部,数据不可靠,直接返回
if ((!BufferData.contains(buff_head))&&(PasteData.isNull())){
return ;
}
//第一种:有头无尾,先清空原有内容,再附加
if ((BufferData.contains(buff_head))&&(!BufferData.contains(buff_end))){
PasteData.clear();
PasteData.append(BufferData);
qDebug()<<"BufferData1:"<<BufferData;
}
//第二种:无头无尾且变量已有内容,数据中段部分,继续附加即可
if ((!BufferData.contains(buff_head))&&(!BufferData.contains(buff_end))&&(!PasteData.isNull())){
PasteData.append(BufferData);
qDebug()<<"BufferData2:"<<BufferData;
}
//第三种:无头有尾且变量已有内容,已完整读取,附加后输出数据,并清空变量
if ((!BufferData.contains(buff_head))&&(BufferData.contains(buff_end))&&(!PasteData.isNull())){
PasteData.append(BufferData);
ReadData = PasteData;
PasteData.clear();
qDebug()<<"BufferData3:"<<BufferData;
}
//第四种:有头有尾(一段完整的内容),先清空原有内容,再附加,然后输出,最后清空变量
if ((BufferData.contains(buff_head))&&(BufferData.contains(buff_end))){
PasteData.clear();
PasteData.append(BufferData);
ReadData = PasteData;
PasteData.clear();
qDebug()<<"BufferData4:"<<BufferData;
}
if(ReadData.endsWith("1}")){
AddData = "}";
ReadData = ReadData.append(AddData);
}
//qDebug()<<"rel data:"<< ReadData.data();
int dataLen = ReadData.length();
if (dataLen <= 0){
qDebug () << "No num!";
}
//在这里分析处理ReadData 即可
ReadData.clear();
}
声明:本文内容由易百纳平台入驻作者撰写,文章观点仅代表作者本人,不代表易百纳立场。如有内容侵权或者其他问题,请联系本站进行删除。
红包
5
5
评论
打赏
- 分享
- 举报
评论
0个
手气红包
暂无数据
相关专栏
-
浏览量:2539次2020-08-14 15:09:41
-
浏览量:4789次2020-08-15 17:29:07
-
浏览量:4827次2020-08-11 18:51:18
-
浏览量:6536次2020-08-15 17:22:26
-
浏览量:4961次2020-08-11 17:39:02
-
浏览量:10265次2021-01-12 20:00:35
-
浏览量:1463次2020-08-30 00:47:44
-
浏览量:10522次2020-08-30 00:41:53
-
浏览量:1456次2020-08-30 00:47:29
-
浏览量:4491次2020-07-31 13:45:09
-
浏览量:1394次2023-11-04 15:07:11
-
浏览量:2403次2020-08-23 21:47:22
-
2020-11-09 12:31:13
-
浏览量:914次2023-10-26 15:18:07
-
浏览量:5276次2020-07-31 11:54:44
-
浏览量:4459次2020-07-27 16:34:42
-
浏览量:3671次2020-07-28 10:49:06
-
2018-06-18 22:47:22
-
浏览量:3745次2020-08-30 10:56:46
置顶时间设置
结束时间
删除原因
-
广告/SPAM
-
恶意灌水
-
违规内容
-
文不对题
-
重复发帖
打赏作者
MacianYuan
您的支持将鼓励我继续创作!
打赏金额:
¥1
¥5
¥10
¥50
¥100
支付方式:
微信支付
打赏成功!
感谢您的打赏,如若您也想被打赏,可前往 发表专栏 哦~
举报反馈
举报类型
- 内容涉黄/赌/毒
- 内容侵权/抄袭
- 政治相关
- 涉嫌广告
- 侮辱谩骂
- 其他
详细说明
审核成功
发布时间设置
发布时间:
请选择发布时间设置
是否关联周任务-专栏模块
审核失败
失败原因
请选择失败原因
备注
请输入备注