ROS:坐标系之间的关系 (map \ odom \ base_link)

ROS:坐标系之间的关系 (map \ odom \ base_link)

在使用ROS进行定位与导航操作时,会伴随着各种坐标系,并且每种坐标系都有明确的含义,ros中定义了常见的坐标系,并且所有的坐标系采用右手坐标系。

1. 三种坐标系

  • base_link : 以机器人自身为原点的坐标系。

​ baselink可以连接到任意位置或方向的本体,且每个硬件平台都有一个不同的位置,并且可以提供明显的参考点。在实际中,我们约定小车base_link的x轴指向机器人前方,y轴为指向小车左方,z轴为指向小车上方。


  • odom: 里程计坐标系

​ odom为一个固定在环境中的坐标系(world fixed),它的原点和方向不会随着机器人运动而改变。但是odom的位置可以随着机器人的运动漂移。漂移导致odom不是一个很有用的长期的全局坐标。然而机器人的odom坐标必须保证是连续变化的。

​ 在odom坐标系下机器人的位置必须是连续变化的,不能有突变和跳跃。在一般使用中odom坐标系是通过里程计信息计算出来的。比如轮子的编码器或者视觉里程计算法或者陀螺仪和加速度计。odom是一个短期的局域的精确坐标系。


  • map : 是一个固定在环境中得世界坐标系

​ 机器人在map坐标系下的坐标不应该随着时间漂移。但是map坐标系下的坐标并不需要保证连续性。map坐标系是一个很有用的长期全局坐标系。但是由于坐标会跳跃改变,这是一个比较差的局部坐标系(不适合用于避障和局部操作)。

可能说到这还是很懵。-__-

可以参考ROS 中的三种坐标系

如果让机器人去所设定的点,机器人需要通过里程计信息去倒推,也表明odom应该与map重合.


2. 坐标系之间的关系

earth ------> map ------ > odom ------> base_link

  • earth : 当有多个机器人的时候,每个机器人都有自己的map坐标系,他们之间的map坐标系并不相同。如果想要在不同的机器人间共享数据,则需要这个earth坐标系来进行转化。
  • mapbase_link通过定位组件计算得出。但是定位组件并不发布从mapbase_link的变换。它首先获取odombase_link的变换然后利用定位信息计算出mapodom的变换。
  • odombase_link的变换由里程计数据源中的一个发布。

​ 他们之间满足如下关系:

base_link + 里程计等计算的信息 -> odom + 矫正(amcl)-> map坐标系

3. add a frame

#include <ros/ros.h>
#include <tf/transform_broadcaster.h>

int main(int argc, char** argv){
  ros::init(argc, argv, "my_tf_broadcaster");
  ros::NodeHandle node;

  tf::TransformBroadcaster br;
  tf::Transform transform;

  //以10hz的速率来循环执行
  ros::Rate rate(10.0);
  while (node.ok()){
    //transform.setOrigin( tf::Vector3(0.0, 2.0, 0.0) ); //静止tf
    transform.setOrigin( tf::Vector3(2.0*sin(ros::Time::now().toSec()), 2.0*cos(ros::Time::now().toSec()), 0.0) );  //随时间变换tf--- x(-2,2) y(-2 ,2)
    std::cout <<"时间:"<<ros::Time::now()<<" "<<ros::Time::now().toSec()<<" "<< sin(ros::Time::now().toSec())<< std::endl;
    
    transform.setRotation( tf::Quaternion(0, 0, 0, 1) );
    br.sendTransform(tf::StampedTransform(transform, ros::Time::now(), "turtle1", "carrot1"));
    rate.sleep();
  }
  return 0;
};
find_package(catkin REQUIRED COMPONENTS
  roscpp
  tf
)

add_executable(frame_tf_broadcaster src/frame_tf_broadcaster.cpp)
target_link_libraries(frame_tf_broadcaster ${catkin_LIBRARIES})

rosrun [add_pkg] frame_tf_broadcaster
参考资料

  1. wiki.ros.org/reps
  2. wiki.ros.org/tf/tutorials(c++)
  3. http://wiki.ros.org/tf/Tutorials
  • 3
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要获取mapbase_link之间的机器人角度,可以使用ROS2中的tf2库。tf2库提供了一个TransformListener类,它可以订阅tf2变换,并提供了一些方法来查询和转换变换。 以下是获取机器人角度的步骤: 1. 创建一个TransformListener对象: ``` tf_buffer = tf2_ros.Buffer() tf_listener = tf2_ros.TransformListener(tf_buffer) ``` 2. 在机器人运行时,订阅tf2中的/map到/base_link变换: ``` try: trans = tf_buffer.lookup_transform('map', 'base_link', rospy.Time()) except (tf2_ros.LookupException, tf2_ros.ConnectivityException, tf2_ros.ExtrapolationException): rospy.logwarn("Failed to get transform from map to base_link") ``` 3. 从变换中获取机器人的角度: ``` roll, pitch, yaw = tf.transformations.euler_from_quaternion([trans.transform.rotation.x, trans.transform.rotation.y, trans.transform.rotation.z, trans.transform.rotation.w]) ``` 其中,roll、pitch和yaw是机器人在x、y和z轴上的旋转角度。 完整的代码示例如下: ``` import rospy import tf2_ros import tf rospy.init_node('tf_listener') tf_buffer = tf2_ros.Buffer() tf_listener = tf2_ros.TransformListener(tf_buffer) while not rospy.is_shutdown(): try: trans = tf_buffer.lookup_transform('map', 'base_link', rospy.Time()) roll, pitch, yaw = tf.transformations.euler_from_quaternion([trans.transform.rotation.x, trans.transform.rotation.y, trans.transform.rotation.z, trans.transform.rotation.w]) rospy.loginfo("Robot angle: %.2f degrees", yaw*180/3.14) except (tf2_ros.LookupException, tf2_ros.ConnectivityException, tf2_ros.ExtrapolationException): rospy.logwarn("Failed to get transform from map to base_link") rospy.sleep(0.1) ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值