news 2026/5/8 3:00:43

【0183转808协议系列】(二)如何用linux C语言实现$GPRMC报文解析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【0183转808协议系列】(二)如何用linux C语言实现$GPRMC报文解析

NMEA 0183协议对RMC消息的定义如下:

我们从串口读取到的模组输出的LOG,有时候可能是好几行数据,所以我们要逐行解析$GPRMC报文。

bytes_read = read_serial(serial_fd, recv_buffer, sizeof(recv_buffer) - 1); if (bytes_read > 0) { recv_buffer[bytes_read] = '\0'; printf("\n\n通过串口从RTK模组读取到原始定位数据:\n%s", recv_buffer); // 解析GPRMC数据并处理JT808协议转换 line = strtok_r(recv_buffer, "\r\n", &saveptr); while (line != NULL) { memset(&gps_data, 0, sizeof(gps_data)); if (parse_gprmc(line, &gps_data) == 0) { printf("\n解析到有效GPS数据:\n纬度=%f, 经度=%f, " "速度=%f, 航向=%f, 日期=%s, 时间=%s\n", gps_data.latitude, gps_data.longitude, gps_data.speed, gps_data.direction, gps_data.date, gps_data.time); ......此处省略代码若干 line = strtok_r(NULL, "\r\n", &saveptr); } }

此处用了strtok_r去分隔字符串而不是用strtok,可以避免原字符串被改变。用while循环去分隔每一行数据,然后对每一行数据进行解析。

解析函数需要实现如下功能:

输入验证‌
首先检查报文是否以$GPRMC开头,确保是GPRMC报文。
使用strncpy复制输入字符串到缓冲区,并确保以\0结尾。

‌字段解析‌
使用strtok_r按逗号分隔报文字段,saveptr保存分隔状态。
可以通过switch语句逐字段解析:
‌时间(HHMMSS)‌:直接复制到gps->time。
‌状态(A/V)‌:存储到gps->status。
‌纬度/经度‌:
为了与JT808协议要求的经纬度格式一致,所以需要将读取到的将度分秒格式(如3104.39321639)的经纬度转换为小数度(如31.073220271666668)。
可以通过atof转换为浮点数,计算分和秒部分。
‌速度/方向‌:直接转换为浮点数存储。
‌日期(DDMMYY)‌:复制到gps->date。

‌错误处理‌
若字段数超过12或非GPRMC报文,返回-1表示解析失败。

具体代码实现如下:

int parse_gprmc(const char *nmea, GPSData *gps) { char buffer[256]; char *token; char *saveptr; // 用于strtok_r的保存指针 int field_count = 0; double lat_sec, lon_sec; int lat_min, lon_min; strncpy(buffer, nmea, sizeof(buffer) - 1); buffer[sizeof(buffer) - 1] = '0'; // 检查是否为GPRMC报文 if (strncmp(buffer, "$GPRMC", 6) != 0) { return -1; } token = strtok_r(buffer, ",", &saveptr); while (token != NULL && field_count < 12) { switch (field_count) { case 1: // 时间 HHMMSS strncpy(gps->time, token, sizeof(gps->time) - 1); break; case 2: // 状态 A/V gps->status = token[0]; break; case 3: // 纬度 if (strlen(token) > 0) { lat_min = (int)(atof(token) / 100); lat_sec = (atof(token) - lat_min * 100); gps->latitude = lat_min + lat_sec / 60.0; } break; case 4: // 纬度半球 N/S // 可根据需要处理 break; case 5: // 经度 if (strlen(token) > 0) { lon_min = (int)(atof(token) / 100); lon_sec = (atof(token) - lon_min * 100); gps->longitude = lon_min + lon_sec / 60.0; } break; case 6: // 经度半球 E/W // 可根据需要处理 break; case 7: // 速度 if (strlen(token) > 0) { gps->speed = atof(token); } break; case 8: // 方向 if (strlen(token) > 0) { gps->direction = atof(token); } break; case 9: // 日期 DDMMYY strncpy(gps->date, token, sizeof(gps->date) - 1); break; } field_count++; token = strtok_r(NULL, ",", &saveptr); } return (gps->status == 'A') ? 0 : -1; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/1 4:18:04

管理案例丨华恒智信助力某电力设计公司绩效考核体系升级项目——以“角色定位”与“多维评价”模型,破解国企业务升级下的价值衡量难题

【客户行业】电力行业、电力工程设计咨询、工程总承包&#xff08;EPC&#xff09;、基础设施建设领域 【问题类型】绩效考核体系优化、转型期组织绩效管理、项目管理与经营类岗位价值评价一、客户背景&#xff1a;从“设计院”到“工程公司”转型中的管理挑战某电力设计公司成…

作者头像 李华
网站建设 2026/4/27 6:44:19

AgentWeb升级实战手册:从架构重构到性能飞跃

AgentWeb升级实战手册&#xff1a;从架构重构到性能飞跃 【免费下载链接】AgentWeb AgentWeb is a powerful library based on Android WebView. 项目地址: https://gitcode.com/gh_mirrors/ag/AgentWeb 还记得那个让你夜不能寐的WebView升级噩梦吗&#xff1f;页面白屏…

作者头像 李华
网站建设 2026/5/1 10:40:34

基于单片机的红外测温系统设计

第一章&#xff1a;系统设计目标与需求分析 本系统旨在通过红外技术实现非接触式温度快速测量&#xff0c;解决传统测温需接触、响应慢的问题&#xff0c;适用于人体体温筛查、设备温升监测等场景。核心需求包括&#xff1a;测温范围设定为-50℃380℃&#xff0c;覆盖多数日常与…

作者头像 李华
网站建设 2026/4/23 2:12:15

使用Postman之上一个接口的返回值作为下一个接口的入参

在使用Postman做接口测试的时候&#xff0c;在多个接口的测试中&#xff0c;如果需要上一个接口的返回值作为下一个接口的入参&#xff0c;其基本思路是&#xff1a; 1、获取上一个接口的返回值 2、将返回值设置成环境变量或者全局变量 3、设置下一个接口的参数形式 下面我…

作者头像 李华
网站建设 2026/5/4 8:32:13

4.6 你的 Agent 安全吗?Prompt 注入、数据泄露攻击模式与防御策略

4.6 你的 Agent 安全吗?Prompt 注入、数据泄露攻击模式与防御策略 导语:在上一章,我们构建了一个“AI 防火墙”,学会了如何检测正在发生的攻击。然而,在安全领域,仅仅满足于“发现问题”是远远不够的。一个真正健壮的系统,需要构建**纵深防御(Defense in Depth)**体系…

作者头像 李华
网站建设 2026/5/2 8:42:08

5.4 模型评估与优化:科学评估微调效果

5.4 模型评估与优化:科学评估微调效果 导语:在上一章的“炼丹”实战中,我们成功地微调出了一个“AI 皮肤科医生”模型,并通过几个简单的对话,直观地感受到了它的变化。但这种“感觉”是主观的、不可靠的。要将“炼丹”从一门“玄学”变为一门“科学”,我们必须引入客观、…

作者头像 李华