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

1 篇文章 0 订阅
1 篇文章 0 订阅

本文记录在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后,显示效果为:
在这里插入图片描述

在MATLAB,可以使用机器人操作系统(ROS)工具箱的函数来将点云转换机器人坐标系下并进行配准。具体步骤如下: 1.将机器人的传感器数据(如激光雷达扫描或RGB-D摄像头)以ROS消息的形式发布到ROS网络。 2.在MATLAB,使用ROS工具箱的函数订阅这些传感器数据,并将其转换为点云格式。 3.使用机器人的运动控制器获取机器人的位姿(位置和方向)信息。 4.使用ROS工具箱的函数将点云从传感器坐标系转换机器人坐标系下,以便它们与机器人的位姿对齐。 5.使用点云配准算法(如Iterative Closest Point,ICP)将点云对齐到机器人坐标系下。 6.将配准后的点云保存或用于下一步的机器人导航或环境建模。 下面是一个简单的MATLAB代码示例,演示如何使用ROS工具箱的函数将点云转换机器人坐标系下并进行配准: ``` % 订阅激光雷达扫描消息 laserSub = rossubscriber('/laser_scan'); % 等待接收到传感器数据 scan = receive(laserSub); % 将激光雷达扫描数据转换为点云格式 ptCloud = pointCloud([scan.Ranges.*cos(scan.AngleMin + (0:359)'*scan.AngleIncrement), ... scan.Ranges.*sin(scan.AngleMin + (0:359)'*scan.AngleIncrement), ... zeros(360,1)]); % 获取机器人的位姿信息 poseSub = rossubscriber('/robot_pose'); pose = receive(poseSub); % 将点云从传感器坐标系转换机器人坐标系下 tfTree = rostf; sensorFrame = 'laser'; robotFrame = 'base_link'; tform = getTransform(tfTree, sensorFrame, robotFrame, 'Timeout', 2); ptCloudRobot = pctransform(ptCloud, affine3d(tform.Transform.Matrix)); % 使用ICP算法将点云对齐到机器人坐标系下 fixedCloud = load('map.mat'); % 加载已知的地图点云 tformICP = pcregrigid(ptCloudRobot, fixedCloud, 'Metric', 'pointToPlane', 'Extrapolate', true); % 显示配准后的点云 ptCloudAligned = pctransform(ptCloudRobot, tformICP); pcshowpair(ptCloudAligned, fixedCloud); ``` 这是一个基本示例,可以根据具体的需求进行修改。需要注意的是,这个示例假设机器人已经在已知的地图定位,并且已经将地图点云加载到变量`fixedCloud`。如果机器人处于未知的环境,需要使用SLAM算法来同时估计机器人的位姿和地图。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Errorbot

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值