GPS学习(一):在ROS2中将GPS经纬度数据转换为机器人ENU坐标系,在RVIZ中显示坐标轨迹

文章目录

  • 一、GPS模块介绍
  • 二、坐标转换
    • 转换原理
      • 参数解释:
    • 增加回调函数
    • 效果演示

本文记录在Ubuntu22.04-Humbel中使用NMEA协议GPS模块的过程,使用国产ROS开发板鲁班猫(LubanCat )进行调试。

一、GPS模块介绍

在淘宝找了款性价比较高的轮趣科技GPS北斗双模定位模块作为入门学习使用,支持GNSS系统(北斗、GPS、GLONASS、日本的QZSS以及卫星增强系统SBAS),定位精度在2.5m左右,属于民用级别。引出了常用的几个外部接口,主要用type-c调试比较方便,售后技术也相当给力。接线示意图如下,注意:测试时需要将GPS天线放置在室外,否则模块将没有GPS信号
在这里插入图片描述

二、坐标转换

机器人基于Ubuntu-ROS平台做开发调试,所以需要将GPS模块的经纬度和高度信息转化为机器人所能认识的本地坐标系(局部笛卡尔坐标系)。参考其他博主案例,使用半正矢公式来将经纬度转为xyz坐标值。

转换原理

半正矢公式介绍:
半正矢公式是一种根据两点的经度和纬度来确定大圆上两点之间距离的计算方法,在导航有着重要地位。它是球面三角学中“半正矢定理”公式的特例,该定理涉及了球面三角形的边和角。
对于任何球面上的两点,圆心角的半正矢值可以通过如下公式计算:

          hav( d r \frac{d}{r} rd)=hav( ψ 2 \psi_{2} ψ2- ψ 1 \psi_{1} ψ1)+cos( ψ 1 \psi_{1} ψ1)cos( ψ 2 \psi_{2} ψ2)hav( λ 2 \lambda_{2} λ2- λ 1 \lambda_{1} λ1)

hav 是半正矢函数的缩写:

          haversin( θ \theta θ)= s i n 2 sin^{2} sin2( ( θ ) 2 \frac{(\theta)}{2} 2(θ))= 1 − c o s ( θ ) 2 \frac{1-cos(\theta)}{2} 21cos(θ)

参数解释:

d :两点之间的距离(沿大圆,见球面距离);
r :球的半径;
ψ 1 \psi_{1} ψ1 ψ 2 \psi_{2} ψ2 :点 1 的纬度和点 2 的纬度,以弧度制度量;
λ 1 \lambda_{1} λ1 λ 2 \lambda_{2} λ2:点 1 的经度和点 2 的经度,以弧度制度量。
等号左边的 d r \frac{d}{r} rd:圆心角,单位是弧度。

所以,可以通过应用反半正矢函数(如果可以查到值)或通过使用反正弦函数来解出d :
           d = r ∗ a r c h a v ( h ) d=\sqrt[]{r*archav(h)} d=rarchav(h)                   =  2r*arcsin h \sqrt[]{h} h )

将h=hav d r \frac{d}{r} rd代入后可得:
   在这里插入图片描述
对于中短距离来说,半正矢公式是计算地球表面 GPS 坐标之间距离的有用且相对准确的方法,在距离较长时其准确性可能会降低。

增加回调函数

这里简单解释一下gpsCallback回调函数的内容,此处订阅sensor_msgs::msg::NavSatFix数据格式的GPS话题进行数据处理。

//将纬度和经度值从度数转换为弧度。
double rad(double d) 
{return d * 3.1415926 / 180.0;
}
void gps_callback(const sensor_msgs::msg::NavSatFix::SharedPtr gps_msg)
{if (!pose_init){// Initialization code remains the same.//使用接收到的 GPS 消息中的纬度、经度和高度false进行初始化,只初始化一次init_pose.latitude = gps_msg->latitude;init_pose.longitude = gps_msg->longitude;init_pose.altitude = gps_msg->altitude;init = true;}else{//计算当前 GPS 位置与初始 GPS 位置之间的纬度和经度变化double radLat1 ,radLat2, radLong1,radLong2,delta_lat,delta_long,x,y;double deltaLat_y ,deltaLong_x;deltaLat_x = 0; deltaLong_y = 0;radLat1 = rad(init_pose.latitude);//初始化位置的纬度radLong1 = rad(init_pose.longitude);//初始化位置的经度radLat2 = rad(gps_msg->latitude);//当前位置的纬度radLong2 = rad(gps_msg->longitude);//当前位置的经度deltaLat = radLat2 - radLat1; deltaLong = radLong2 - radLong1;// 代入公式进行计算x的坐标值x = 2 * asin(sqrt(pow(sin(delta_lat / 2), 2) + cos(radLat1) * cos(radLat2) * pow(sin(deltaLong_y / 2), 2));//公式计算结果默认为弧度,所以这里要转换成m的单位x = x * EARTH_RADIUS * 1000; //  代入公式进行计算y的坐标值y = 2 * asin(sqrt(pow(sin(deltaLat_x / 2), 2) + cos(radLat2) * cos(radLat2) * pow(sin(deltaLong / 2), 2));y = y * EARTH_RADIUS * 1000; // Convert to meters.// 将高度差计算为z的坐标值double z = gps_msg->altitude - init_pose.altitude;//将计算得到的xyz值发布为ROS中的PoseStamped数据,作为路径发布ros_path_.header.frame_id = "path";ros_path_.header.stamp = rclcpp::Node::now();geometry_msgs::msg::PoseStamped pose;pose.header = ros_path_.header;pose.pose.position.x = x;pose.pose.position.y = y;pose.pose.position.z = z;ros_path_.poses.push_back(pose);state_pub_.publish(ros_path_);}
}

效果演示

使用nmea_navsat_driver驱动启动GPS模块,源码链接:https://github.com/ros-drivers/nmea_navsat_driver/tree/ros2

ros2 launch nmea_navsat_driver nmea_serial_driver.launch.py

然后启动坐标转换节点,将经纬度数据从WGS-84 坐标转换到机器人真实世
界 xyz 坐标系下(一般为东北天方向),以第一个经纬度数据为原点,不断描绘实时经纬度数据的位移方向,显示为机器人运行轨迹图。

ros2 launch gps_path gps_path.launch.py

打开rviz2,修改 Fixed Frame 为path,电机左下角[add],根据topic添加[gps_path]->path后,显示效果为:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/155414.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LLM - 训练与推理过程中的 GPU 算力评估

目录 一.引言 二.FLOPs 和 TFLOPs ◆ FLOPs [Floating point Opearation Per Second] ◆ TFLOPs [Tera Floating point Opearation Per Second] 三.训练阶段的 GPU 消耗 ◆ 影响训练的因素 ◆ GPT-3 训练统计 ◆ 自定义训练 GPU 评估 四.推理阶段的 GPU 消耗 ◆ 影响…

识别flink的反压源头

背景 flink中最常见的问题就是反压,这种情况下我们要正确的识别导致反压的真正的源头,本文就简单看下如何正确识别反压的源头 反压的源头 首先我们必须意识到现实中轻微的反压是没有必要去优化的,因为这种情况下是由于偶尔的流量峰值,Task…

PHP连接SQLServer echo输出中文汉字显示乱码解决方法

1、查询结果有中文会显示乱码。 解决方法一&#xff08;较简单&#xff0c;建议使用&#xff09;&#xff1a; 在php文件最开头写上&#xff1a; header(Content-type: text/html; charsetUTF8); // UTF8不行改成GBK试试&#xff0c;与你保存的格式匹配 <?php header(&q…

python 练习 在列表元素中合适的位置插入 输入值

目的&#xff1a; 有一列从小到大排好的数字元素列表&#xff0c; 现在想往其插入一个值&#xff0c;要求&#xff1a; 大于右边数字小于左边数字 列表元素&#xff1a; [1,4,6,13,16,19,28,40,100] # 方法&#xff1a; 往列表中添加一个数值&#xff0c;其目的方便元素位置往后…

APISpace 全国快递物流地图轨迹查询API接口案例代码

1.全国快递物流地图轨迹查询接口详解 1.1 接口请求 请求方式&#xff1a;POST请求地址&#xff1a;https://eolink.o.apispace.com/wldtgj1/paidtobuy_api/trace_map请求头&#xff1a; 标签必填说明X-APISpace-Token是鉴权私钥&#xff0c;登陆 APISpace 后在管理后台的[访…

对xss-labs靶场的一次XSS攻击

1、首先我们进入靶场&#xff0c;提示我们开始测试 2、我使用AWVS工具进行了先行扫描&#xff0c;发现爆出XSS漏洞 3、然后对症下药 在输入框中输入&#xff1a; <script>alert(document.cookie)</script> 4、进入下一关 5、我们直接执行<script>…

【c++|opencv】一、基础操作---2.图像信息获取

every blog every motto: You can do more than you think. https://blog.csdn.net/weixin_39190382?typeblog 0. 前言 图像信息获取&#xff0c;roi 1. 图像信息获取 // 获取图像信息#include <iostream> #include <opencv2/opencv.hpp>using namespace cv; …

【Docker】从命名空间和路由角度探究Docker的bridge网络

桥接网络是Docker的默认网络模式。在桥接网络中&#xff0c;Docker会为每个容器创建一个虚拟网络接口&#xff0c;并为容器分配一个IP地址。容器可以通过桥接网络与主机和其他容器进行通信&#xff0c;也能暴露端口供外部访问。 容器之间的通信原理 首先我们创建两个容器&…

三.RocketMQ单机安装及集群搭建

RocketMQ单机安装及集群搭建 一&#xff1a;安装环境1.软硬件要求2.下载RocketMQ 二.安装单机MQ1.上传并解压2.目录介绍3.修改MQ启动时初始JVM内存4.启动NameServer与Broker5.测试RocketMQ 三.RocketMQ集群搭建1.集群概念特点2.集群模式分类3.集群工作流程4.双主双从集群搭建4.…

git本地搭建服务器[Vmware虚拟机访问window的git服务器]

先按照https://zhuanlan.zhihu.com/p/494988089说明下载好Gitblit然后复制到tomcat的webapps目录下,如下: 双击"startup.bat"启动tomcat: 然后访问"http://127.0.0.1:8080/gitblit/"即可看到git的界面: 说明git服务器已经能够成功运行了! Vmware虚拟机…

Linux的简介和环境搭建

简介 Linux是一套免费使用和自由传播的类Unix操作系统&#xff0c;是一个基于POSIX和Unix的多用户、多任务、支持多线程和多CPU的操作系统。它能运行主要的Unix工具软件、应用程序和网络协议。它支持32位和64位硬件。Linux继承了Unix以网络为核心的设计思想&#xff0c;是一个…

5000张照片怎么快速发给别人?分享三个简单的方法!

有的时候我们不得不一次性发送很多图片&#xff0c;一张一张发实在让人头疼&#xff0c;这个时候就需要借助一些图片压缩工具打包成文件压缩包发送。下面介绍了三种好用的方法&#xff0c;一起来看看吧&#xff5e; 方法一&#xff1a;使用微信助手 可以使用微信助手&#xff…