在ROS中内置一些比较实用的工具,通过这些工具可以方便快捷的实现某个功能或调试程序,从而提高开发效率,本章主要介绍ROS中内置的如下组件:

  • TF坐标变换,实现不同类型的坐标系之间的转换;
  • rosbag 用于录制ROS节点的执行过程并可以重放该过程;
  • rqt 工具箱,集成了多款图形化的调试工具。

本章预期达成的学习目标:

  • 了解 TF 坐标变换的概念以及应用场景;
  • 能够独立完成TF案例:小乌龟跟随;
  • 可以使用 rosbag 命令或编码的形式实现录制与回放;
  • 能够熟练使用rqt中的图形化工具。

案例演示: **小乌龟跟随实现,该案例是ros中内置案例**,终端下键入启动命令

  1. roslaunch turtle_tf2 turtle_tf2_demo_cpp.launchroslaunch turtle_tf2 turtle_tf2_demo.launch

键盘可以控制一只乌龟运动,另一只跟随运动。

1、TF坐标换

机器人系统上,有多个传感器,如激光雷达、摄像头等,有的传感器是可以感知机器人周边的物体方位(或者称之为:坐标,横向、纵向、高度的距离信息)的,以协助机器人定位障碍物,可以直接将物体相对该传感器的方位信息,等价于物体相对于机器人系统或机器人其它组件的方位信息吗?显示是不行的,这中间需要一个转换过程。更具体描述如下:
场景1:雷达与小车 现有一移动式机器人底盘,在底盘上安装了一雷达,雷达相对于底盘的偏移量已知,现雷达检测到一障碍物信息,获取到坐标分别为(x,y,z),该坐标是以雷达为参考系的,如何将这个坐标转换成以小车为参考系的坐标呢?

第五章 ROS常用组件. - 图2

场景2:现有一带机械臂的机器人(比如:PR2)需要夹取目标物,当前机器人头部摄像头可以探测到目标物的坐标(x,y,z),不过该坐标是以摄像头为参考系的,而实际操作目标物的是机械臂的夹具,当前我们需要将该坐标转换成相对于机械臂夹具的坐标,这个过程如何实现?

第五章 ROS常用组件. - 图3

当然,根据我们高中学习的知识,在明确了不同坐标系之间的的相对关系,就可以实现任何坐标点在不同坐标系之间的转换,但是该计算实现是较为常用的,且算法也有点复杂,因此在 ROS 中直接封装了相关的模块: 坐标变换(TF)。

概念

tf:TransForm Frame,坐标变换

坐标系:ROS 中是通过坐标系统开标定物体的,确切的将是通过右手坐标系来标定的。

第五章 ROS常用组件. - 图4

作用

在 ROS 中用于实现不同坐标系之间的点或向量的转换。

案例

小乌龟跟随案例:如本章引言部分演示。

说明

在ROS中坐标变换最初对应的是tf,不过在 hydro 版本开始**, tf 被弃用迁移到 tf2**,后者更为简洁高效,tf2对应的常用功能包有:

tf2_geometry_msgs:可以将ROS消息转换成tf2消息。

tf2: 封装了坐标变换的常用消息。

tf2_ros:为tf2提供了roscpp和rospy绑定,封装了坐标变换常用的API。

1、坐标msg消息

订阅发布模型中数据载体 msg 是一个重要实现,首先需要了解一下,在坐标转换实现中常用的msg

  1. geometry_msgs/TransformStamped //坐标系相关的位置消息
  1. 或者geometry_msgs/PointStamped //坐标点相关的位置消息

前者用于传输**坐标系相关位置信息,后者用于传输某个坐标系内坐标点的信息**。在坐标变换中,频繁的需要使用到坐标系的相对关系以及坐标点信息。

1.geometry_msgs/TransformStamped(传输**坐标系相关位置信息**

命令行键入:
  1. rosmsg info geometry_msgs/TransformStamped
  1. std_msgs/Header header #头信息
  2. uint32 seq #|-- 序列号
  3. time stamp #|-- 时间戳
  4. string frame_id #|-- 坐标 ID
  5. string child_frame_id #子坐标系的 id
  6. geometry_msgs/Transform transform #坐标信息
  7. geometry_msgs/Vector3 translation #偏移量
  8. float64 x #|-- X 方向的偏移量
  9. float64 y #|-- Y 方向的偏移量
  10. float64 z #|-- Z 方向上的偏移量
  11. geometry_msgs/Quaternion rotation #四元数
  12. float64 x
  13. float64 y
  14. float64 z
  15. float64 w

四元数用于表示坐标的相对姿态

第五章 ROS常用组件. - 图5

2.geometry_msgs/PointStamped(传输某个坐标系内坐标点的信息

命令行键入:
  1. rosmsg info geometry_msgs/PointStamped
  1. std_msgs/Header header #头
  2. uint32 seq #|-- 序号
  3. time stamp #|-- 时间戳
  4. string frame_id #|-- 所属坐标系的 id
  5. geometry_msgs/Point point #点坐标
  6. float64 x #|-- x y z 坐标
  7. float64 y
  8. float64 z

第五章 ROS常用组件. - 图6

2、静态坐标

所谓静态坐标变换,是指两个坐标系之间的相对位置是固定的。

需求描述:

现有一机器人模型,核心构成包含主体与雷达,各对应一坐标系,坐标系的原点分别位于主体与雷达的物理中心,已知雷达原点相对于主体原点位移关系如下: x 0.2 y0.0 z0.5。当前雷达检测到一障碍物,在雷达坐标系中障碍物的坐标为 (2.0 3.0 5.0),请问,该障碍物相对于主体的坐标是多少?

结果演示:

第五章 ROS常用组件. - 图7

第五章 ROS常用组件. - 图8

实现分析:

  1. 坐标系相对关系,可以通过发布方发布
  2. 订阅方,订阅到发布的坐标系相对关系,再传入坐标点信息(可以写死),然后借助于 tf 实现坐标变换,并将结果输出

实现流程:C++ 与 Python 实现流程一致

  1. 新建功能包,添加依赖
  2. 编写发布方实现
  3. 编写订阅方实现
  4. 执行并查看结果

方案A:C++实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs

2.发布方(发布雷达坐标系的位置信息)

  1. /*
  2. 静态坐标系变换发布方:
  3. 发布关于 laser 坐标系的位置信息 准确的来说是发布雷达坐标系相对于车体坐标系的位置信息。
  4. (供订阅方的雷达得到周围相对于雷达坐标系的坐标转换成相对于车体的坐标来使用)
  5. 实现流程:
  6. 1.包含头文件
  7. 2.初始化 ROS 节点
  8. 3.创建静态坐标转换广播器
  9. 4.创建坐标系信息
  10. 5.广播器发布坐标系信息
  11. 6.spin()
  12. */
  13. // 1.包含头文件
  14. #include "ros/ros.h"
  15. #include "tf2_ros/static_transform_broadcaster.h" //广播
  16. #include "geometry_msgs/TransformStamped.h" //坐标系
  17. #include "tf2/LinearMath/Quaternion.h" //四元数头文件
  18. int main(int argc, char *argv[])
  19. {
  20. setlocale(LC_ALL,"");
  21. // 2.初始化 ROS 节点
  22. ros::init(argc,argv,"static_brocast");
  23. // 3.创建静态坐标转换广播器 (实例化对象)
  24. tf2_ros::StaticTransformBroadcaster broadcaster;
  25. // 4.创建坐标系信息(实例化对象)
  26. geometry_msgs::TransformStamped ts;
  27. //----设置头信息
  28. ts.header.seq = 100; //序列号
  29. ts.header.stamp = ros::Time::now(); //时间戳
  30. ts.header.frame_id = "base_link"; //父(车体)坐标系 ID 很关键
  31. //----设置子级坐标系
  32. ts.child_frame_id = "laser"; //子(雷达)坐标系ID 很关键
  33. //----设置子级相对于父级的偏移量 (注音是谁相对与谁)(雷达相对于车体)(实际中可能需要测量或者标定)
  34. ts.transform.translation.x = 0.2;
  35. ts.transform.translation.y = 0.0;
  36. ts.transform.translation.z = 0.5;
  37. //----设置四元数:将 欧拉角数据转换成四元数
  38. tf2::Quaternion qtn; //实例化四元数对象,这里假设没有偏角,已经对齐(用惯导的术语来说就是只有杆臂误差,没有失准角)
  39. qtn.setRPY(0,0,0);
  40. ts.transform.rotation.x = qtn.getX();
  41. ts.transform.rotation.y = qtn.getY();
  42. ts.transform.rotation.z = qtn.getZ();
  43. ts.transform.rotation.w = qtn.getW();
  44. // 5.广播器发布坐标系信息 ,将ts这个坐标对象发布广播出去
  45. broadcaster.sendTransform(ts);
  46. ros::spin();//惯例
  47. return 0;
  48. }

3.订阅方

  1. /*
  2. 订阅发布方发布的坐标系关系的信息,生成一个相对于 子级坐标系的坐标点数据,转换成父级坐标系中的坐标点
  3. (得到相对于子级雷达坐标系的数据)——>(转化成相对于车体坐标系的坐标数据)
  4. 实现流程:
  5. 1.包含头文件
  6. 2.初始化 ROS 节点
  7. 3.创建 TF 订阅节点
  8. 4.生成一个坐标点(相对于子级坐标系)
  9. 5.转换坐标点(相对于父级坐标系)
  10. 6.spin()
  11. */
  12. //1.包含头文件
  13. #include "ros/ros.h"
  14. #include "tf2_ros/transform_listener.h" //收听
  15. #include "tf2_ros/buffer.h" //
  16. #include "geometry_msgs/PointStamped.h" //坐标点
  17. #include "tf2_geometry_msgs/tf2_geometry_msgs.h" //注意: 调用 transform 必须包含该头文件
  18. int main(int argc, char *argv[])
  19. {
  20. setlocale(LC_ALL,"");
  21. // 2.初始化 ROS 节点
  22. ros::init(argc,argv,"tf_sub");
  23. ros::NodeHandle nh;
  24. // 3.创建 TF 订阅节点 实例化对象
  25. tf2_ros::Buffer buffer;
  26. tf2_ros::TransformListener listener(buffer);
  27. ros::Rate r(1); //设置频率
  28. while (ros::ok())
  29. {
  30. // 4.生成一个坐标点(相对于子级坐标系,在雷达坐标系下)
  31. geometry_msgs::PointStamped point_laser; //实例化坐标点对象point_laser
  32. point_laser.header.frame_id = "laser"; //所属坐标系的ID
  33. point_laser.header.stamp = ros::Time::now(); //时间戳
  34. point_laser.point.x = 1;
  35. point_laser.point.y = 2;
  36. point_laser.point.z = 7.3;
  37. // 5.转换坐标点(相对于父级车体坐标系)
  38. //新建一个坐标点,用于接收转换结果
  39. //--------------使用 try 语句或休眠,否则可能由于缓存接收延迟而导致坐标转换失败------------------------
  40. try
  41. {
  42. geometry_msgs::PointStamped point_base; //新建坐标点对象
  43. point_base = buffer.transform(point_laser,"base_link"); //关键在于父坐标系的IDbase_link需要和前面定义的一致
  44. ROS_INFO("转换后的数据:(%.2f,%.2f,%.2f),参考的坐标系是:",point_base.point.x,point_base.point.y,point_base.point.z,point_base.header.frame_id.c_str());
  45. }
  46. catch(const std::exception& e)
  47. {
  48. // std::cerr << e.what() << '\n';
  49. ROS_INFO("程序异常.....");
  50. }
  51. r.sleep();
  52. ros::spinOnce(); //一般操作
  53. }
  54. return 0;
  55. }

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,控制台将输出,坐标转换后的结果。
方案B:Python实现

1.创建功能包(不管是c++还是py一般创建项目需要依赖的功能包一致即可)(另外:对于后来增添功能包的完成注意点现在还没整理,不过应该就在那几个文件的地方修改即可)

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs

2.发布方

  1. #! /usr/bin/env python
  2. """
  3. 静态坐标变换发布方:
  4. 发布关于 laser 坐标系的位置信息
  5. 实现流程:
  6. 1.导包
  7. 2.初始化 ROS 节点
  8. 3.创建 静态坐标广播器
  9. 4.创建并组织被广播的消息
  10. 5.广播器发送消息
  11. 6.spin
  12. """
  13. # 1.导包
  14. import rospy
  15. import tf2_ros
  16. import tf
  17. from geometry_msgs.msg import TransformStamped
  18. if __name__ == "__main__":
  19. # 2.初始化 ROS 节点
  20. rospy.init_node("static_tf_pub_p")
  21. # 3.创建 静态坐标广播器
  22. broadcaster = tf2_ros.StaticTransformBroadcaster()
  23. # 4.创建并组织被广播的消息,实例化对象
  24. tfs = TransformStamped()
  25. # --- 头信息
  26. tfs.header.frame_id = "world" //父坐标系ID
  27. tfs.header.stamp = rospy.Time.now() //时间戳
  28. tfs.header.seq = 101 //序列号
  29. # --- 子坐标系
  30. tfs.child_frame_id = "radar" //子坐标系ID
  31. # --- 坐标系相对信息
  32. # ------ 偏移量(杆臂)
  33. tfs.transform.translation.x = 0.2
  34. tfs.transform.translation.y = 0.0
  35. tfs.transform.translation.z = 0.5
  36. # ------ 四元数(失准角)
  37. qtn = tf.transformations.quaternion_from_euler(0,0,0)
  38. tfs.transform.rotation.x = qtn[0]
  39. tfs.transform.rotation.y = qtn[1]
  40. tfs.transform.rotation.z = qtn[2]
  41. tfs.transform.rotation.w = qtn[3]
  42. # 5.广播器发送消息
  43. broadcaster.sendTransform(tfs)
  44. # 6.spin
  45. rospy.spin()
权限设置以及配置文件此处略。

3.订阅方

  1. #! /usr/bin/env python
  2. """
  3. 订阅坐标系信息,生成一个相对于 子级坐标系的坐标点数据,
  4. 转换成父级坐标系中的坐标点
  5. 实现流程:
  6. 1.导包
  7. 2.初始化 ROS 节点
  8. 3.创建 TF 订阅对象
  9. 4.创建一个 radar 坐标系中的坐标点
  10. 5.调研订阅对象的 API 将 4 中的点坐标转换成相对于 world 的坐标
  11. 6.spin
  12. """
  13. # 1.导包
  14. import rospy
  15. import tf2_ros
  16. # 不要使用 geometry_msgs,需要使用 tf2 内置的消息类型
  17. from tf2_geometry_msgs import PointStamped
  18. # from geometry_msgs.msg import PointStamped
  19. if __name__ == "__main__":
  20. # 2.初始化 ROS 节点
  21. rospy.init_node("static_sub_tf_p")
  22. # 3.创建 TF 订阅对象
  23. buffer = tf2_ros.Buffer()
  24. listener = tf2_ros.TransformListener(buffer)
  25. rate = rospy.Rate(1)
  26. while not rospy.is_shutdown():
  27. # 4.创建一个 radar 坐标系中的坐标点
  28. point_source = PointStamped() #实例化对象point_source
  29. point_source.header.frame_id = "radar" //本坐标系ID
  30. point_source.header.stamp = rospy.Time.now() //时间戳
  31. point_source.point.x = 10
  32. point_source.point.y = 2
  33. point_source.point.z = 3
  34. try:
  35. # 5.调研订阅对象的 API 将 4 中的点坐标转换成相对于 world 的坐标
  36. point_target = buffer.transform(point_source,"world") //进行转换
  37. rospy.loginfo("转换结果:x = %.2f, y = %.2f, z = %.2f",
  38. point_target.point.x,
  39. point_target.point.y,
  40. point_target.point.z)
  41. except Exception as e:
  42. rospy.logerr("异常:%s",e)
  43. # 6.spin
  44. rate.sleep()
权限设置以及配置文件此处略。

PS: 在 tf2 的 python 实现中,tf2 已经封装了一些消息类型,不可以使用 geometry_msgs.msg 中的类型

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,控制台将输出,坐标转换后的结果。

第五章 ROS常用组件. - 图9


补充1:(相对静态直接发布即可,不需要写单独的发布程序代码)

当坐标系之间的相对位置固定时,那么所需参数也是固定的: 父系坐标名称、子级坐标系名称、x偏移量、y偏移量、z偏移量、x 翻滚角度、y俯仰角度、z偏航角度,实现逻辑相同,参数不同,那么 ROS 系统就已经封装好了专门的节点,使用方式如下:
  1. rosrun tf2_ros static_transform_publisher x偏移量 y偏移量 z偏移量 z偏航角度 y俯仰角度 x翻滚角度 父级坐标系 子级坐标系

示例:

  1. rosrun tf2_ros static_transform_publisher 0.2 0 0.5 0 0 0 /baselink /laser

也建议使用该种方式直接实现静态坐标系相对信息发布。

补充2:

可以借助于rviz显示坐标系关系,具体操作:(现在还没有接触Rviz)

  • 新建窗口输入命令:rviz;
  • 在启动的 rviz 中设置Fixed Frame 为** base_link**;
  • 点击左下的 add 按钮,在弹出的窗口中选择 TF 组件,即可显示坐标关系。

3、动态坐标变换

  1. **<font style="color:#F5222D;">所谓动态坐标变换,是指两个坐标系之间的相对位置是变化的。</font>**

需求描述:

启动 turtlesim_node,该节点中窗体有一个世界坐标系(左下角为坐标系原点),乌龟是另一个坐标系,键盘控制乌龟运动,将两个坐标系的相对位置动态发布

结果演示:

第五章 ROS常用组件. - 图10

实现分析:

  1. 乌龟本身不但可以看作坐标系,也是世界坐标系中的一个坐标点
  2. 订阅 turtle1/pose,可以获取乌龟在世界坐标系的 x坐标、y坐标、偏移量以及线速度和角速度
  3. 将 pose 信息转换成 坐标系相对信息并发布

实现流程:C++ 与 Python 实现流程一致

  1. 1. <font style="color:#333333;">新建功能包,添加依赖</font>
  2. 2. <font style="color:#333333;">创建坐标相对关系发布方(同时需要订阅乌龟位姿信息)</font>
  3. 3. <font style="color:#333333;">创建坐标相对关系订阅方</font>
  4. 4. <font style="color:#333333;">执行</font>
方案A: C++实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.发布方

  1. /*
  2. 动态的坐标系相对姿态发布(一个坐标系相对于另一个坐标系的相对姿态是不断变动的)
  3. 需求: 启动 turtlesim_node,该节点中窗体有一个世界坐标系(左下角为坐标系原点),乌龟是另一个坐标系,键盘
  4. 控制乌龟运动,将两个坐标系的相对位置动态发布
  5. 实现分析:
  6. 1.乌龟本身不但可以看作坐标系,也是世界坐标系中的一个坐标点
  7. 2.订阅 turtle1/pose,可以获取乌龟在世界坐标系的 x坐标、y坐标、偏移量以及线速度和角速度
  8. 3. pose 信息转换成 坐标系相对信息并发布
  9. 实现流程:
  10. 1.包含头文件
  11. 2.初始化 ROS 节点
  12. 3.创建 ROS 句柄
  13. 4.创建订阅对象
  14. 5.回调函数处理订阅到的数据(实现TF广播)
  15. 5-1.创建 TF 广播器
  16. 5-2.创建 广播的数据(通过 pose 设置)
  17. 5-3.广播器发布数据
  18. 6.spin
  19. */
  20. // 1.包含头文件
  21. #include "ros/ros.h"
  22. #include "turtlesim/Pose.h"
  23. #include "tf2_ros/transform_broadcaster.h"
  24. #include "geometry_msgs/TransformStamped.h"
  25. #include "tf2/LinearMath/Quaternion.h"
  26. //ConstPtr& pose 常量指针的形式
  27. void doPose(const turtlesim::Pose::ConstPtr& pose){
  28. // 5-1.创建 TF 广播器(实例化广播对象)
  29. static tf2_ros::TransformBroadcaster broadcaster;
  30. // 5-2.创建 广播的数据(通过 pose 设置)(实例化对象)
  31. geometry_msgs::TransformStamped tfs;
  32. // |----头设置
  33. tfs.header.frame_id = "world"; //父坐标系ID
  34. tfs.header.stamp = ros::Time::now();
  35. // |----坐标系 ID
  36. tfs.child_frame_id = "turtle1"; //子坐标系ID
  37. // |----坐标系相对信息设置
  38. tfs.transform.translation.x = pose->x;
  39. tfs.transform.translation.y = pose->y;
  40. tfs.transform.translation.z = 0.0; // 二维实现,pose 中没有zz 0
  41. // |--------- 四元数设置,只有航向角信息
  42. tf2::Quaternion qtn;
  43. qtn.setRPY(0,0,pose->theta);
  44. tfs.transform.rotation.x = qtn.getX();
  45. tfs.transform.rotation.y = qtn.getY();
  46. tfs.transform.rotation.z = qtn.getZ();
  47. tfs.transform.rotation.w = qtn.getW();
  48. // 5-3.广播器发布数据
  49. broadcaster.sendTransform(tfs);
  50. }
  51. int main(int argc, char *argv[])
  52. {
  53. setlocale(LC_ALL,"");
  54. // 2.初始化 ROS 节点
  55. ros::init(argc,argv,"dynamic_tf_pub");
  56. // 3.创建 ROS 句柄
  57. ros::NodeHandle nh;
  58. // 4.创建订阅对象(先订阅乌龟的位姿信息)(注意注意乌龟位姿消息对应的话题名称要写对)(回调函数)
  59. ros::Subscriber sub = nh.subscribe<turtlesim::Pose>("/turtle1/pose",1000,doPose);
  60. // 5.回调函数处理订阅到的数据(实现TF广播)
  61. //
  62. // 6.spin
  63. ros::spin();
  64. return 0;
  65. }

3.订阅方

  1. //1.包含头文件
  2. #include "ros/ros.h"
  3. #include "tf2_ros/transform_listener.h"
  4. #include "tf2_ros/buffer.h"
  5. #include "geometry_msgs/PointStamped.h"
  6. #include "tf2_geometry_msgs/tf2_geometry_msgs.h" //注意: 调用 transform 必须包含该头文件
  7. int main(int argc, char *argv[])
  8. {
  9. setlocale(LC_ALL,"");
  10. // 2.初始化 ROS 节点
  11. ros::init(argc,argv,"dynamic_tf_sub");
  12. ros::NodeHandle nh;
  13. // 3.创建 TF 订阅节点
  14. tf2_ros::Buffer buffer; //实例化对象
  15. tf2_ros::TransformListener listener(buffer);
  16. ros::Rate r(1);
  17. while (ros::ok())
  18. {
  19. // 4.生成一个坐标点(相对于子级坐标系)
  20. geometry_msgs::PointStamped point_laser; //实例化对象
  21. point_laser.header.frame_id = "turtle1";
  22. point_laser.header.stamp = ros::Time();
  23. point_laser.point.x = 1;
  24. point_laser.point.y = 1;
  25. point_laser.point.z = 0;
  26. // 5.转换坐标点(相对于父级坐标系)
  27. //新建一个坐标点,用于接收转换结果
  28. //--------------使用 try 语句或休眠,否则可能由于缓存接收延迟而导致坐标转换失败------------------------
  29. try
  30. {
  31. geometry_msgs::PointStamped point_base; //实例化对象
  32. point_base = buffer.transform(point_laser,"world");
  33. ROS_INFO("坐标点相对于 world 的坐标为:(%.2f,%.2f,%.2f)",point_base.point.x,point_base.point.y,point_base.point.z);
  34. }
  35. catch(const std::exception& e)
  36. {
  37. // std::cerr << e.what() << '\n';
  38. ROS_INFO("程序异常:%s",e.what());
  39. }
  40. r.sleep();
  41. ros::spinOnce();
  42. }
  43. return 0;
  44. }
配置文件此处略。

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,与演示结果类似。 可以使用 rviz 查看坐标系相对关系。
方案B:Python实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.发布方

  1. #! /usr/bin/env python
  2. """
  3. 动态的坐标系相对姿态发布(一个坐标系相对于另一个坐标系的相对姿态是不断变动的)
  4. 需求: 启动 turtlesim_node,该节点中窗体有一个世界坐标系(左下角为坐标系原点),乌龟是另一个坐标系,键盘
  5. 控制乌龟运动,将两个坐标系的相对位置动态发布
  6. 实现分析:
  7. 1.乌龟本身不但可以看作坐标系,也是世界坐标系中的一个坐标点
  8. 2.订阅 turtle1/pose,可以获取乌龟在世界坐标系的 x坐标、y坐标、偏移量以及线速度和角速度
  9. 3.将 pose 信息转换成 坐标系相对信息并发布
  10. 实现流程:
  11. 1.导包
  12. 2.初始化 ROS 节点
  13. 3.订阅 /turtle1/pose 话题消息
  14. 4.回调函数处理
  15. 4-1.创建 TF 广播器
  16. 4-2.创建 广播的数据(通过 pose 设置)
  17. 4-3.广播器发布数据
  18. 5.spin
  19. """
  20. # 1.导包
  21. import rospy
  22. import tf2_ros
  23. import tf
  24. from turtlesim.msg import Pose
  25. from geometry_msgs.msg import TransformStamped
  26. # 4.回调函数处理
  27. def doPose(pose):
  28. # 4-1.创建 TF 广播器 //实例化对象
  29. broadcaster = tf2_ros.TransformBroadcaster()
  30. # 4-2.创建 广播的数据(通过 pose 设置) //实例化对象
  31. tfs = TransformStamped()
  32. tfs.header.frame_id = "world"
  33. tfs.header.stamp = rospy.Time.now()
  34. tfs.child_frame_id = "turtle1"
  35. tfs.transform.translation.x = pose.x
  36. tfs.transform.translation.y = pose.y
  37. tfs.transform.translation.z = 0.0
  38. qtn = tf.transformations.quaternion_from_euler(0,0,pose.theta)
  39. tfs.transform.rotation.x = qtn[0]
  40. tfs.transform.rotation.y = qtn[1]
  41. tfs.transform.rotation.z = qtn[2]
  42. tfs.transform.rotation.w = qtn[3]
  43. # 4-3.广播器发布数据
  44. broadcaster.sendTransform(tfs)
  45. if __name__ == "__main__":
  46. # 2.初始化 ROS 节点
  47. rospy.init_node("dynamic_tf_pub_p")
  48. # 3.订阅 /turtle1/pose 话题消息
  49. sub = rospy.Subscriber("/turtle1/pose",Pose,doPose)
  50. # 4.回调函数处理
  51. # 4-1.创建 TF 广播器
  52. # 4-2.创建 广播的数据(通过 pose 设置)
  53. # 4-3.广播器发布数据
  54. # 5.spin
  55. rospy.spin()

3.订阅方

  1. #! /usr/bin/env python
  2. """
  3. 动态的坐标系相对姿态发布(一个坐标系相对于另一个坐标系的相对姿态是不断变动的)
  4. 需求: 启动 turtlesim_node,该节点中窗体有一个世界坐标系(左下角为坐标系原点),乌龟是另一个坐标系,键盘
  5. 控制乌龟运动,将两个坐标系的相对位置动态发布
  6. 实现分析:
  7. 1.乌龟本身不但可以看作坐标系,也是世界坐标系中的一个坐标点
  8. 2.订阅 turtle1/pose,可以获取乌龟在世界坐标系的 x坐标、y坐标、偏移量以及线速度和角速度
  9. 3.将 pose 信息转换成 坐标系相对信息并发布
  10. 实现流程:
  11. 1.导包
  12. 2.初始化 ROS 节点
  13. 3.创建 TF 订阅对象
  14. 4.处理订阅的数据
  15. """
  16. # 1.导包
  17. import rospy
  18. import tf2_ros
  19. # 不要使用 geometry_msgs,需要使用 tf2 内置的消息类型
  20. from tf2_geometry_msgs import PointStamped
  21. # from geometry_msgs.msg import PointStamped
  22. if __name__ == "__main__":
  23. # 2.初始化 ROS 节点
  24. rospy.init_node("static_sub_tf_p")
  25. # 3.创建 TF 订阅对象
  26. buffer = tf2_ros.Buffer()
  27. listener = tf2_ros.TransformListener(buffer)
  28. rate = rospy.Rate(1)
  29. while not rospy.is_shutdown():
  30. # 4.创建一个 radar 坐标系中的坐标点
  31. point_source = PointStamped() #实例化对象point_source
  32. point_source.header.frame_id = "turtle1"
  33. point_source.header.stamp = rospy.Time.now()
  34. point_source.point.x = 10
  35. point_source.point.y = 2
  36. point_source.point.z = 3
  37. try:
  38. # 5.调研订阅对象的 API 将 4 中的点坐标转换成相对于 world 的坐标 新的点对象point_target
  39. point_target = buffer.transform(point_source,"world",rospy.Duration(1))
  40. rospy.loginfo("转换结果:x = %.2f, y = %.2f, z = %.2f",
  41. point_target.point.x,
  42. point_target.point.y,
  43. point_target.point.z)
  44. except Exception as e:
  45. rospy.logerr("异常:%s",e)
  46. # 6.spin
  47. rate.sleep()
权限设置以及配置文件此处略。

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,与演示结果类似。 可以使用 rviz 查看坐标系相对关系。

4、多坐标换(中间媒介坐标变换)

需求描述:

现有坐标系统,父级坐标系统 world,下有两子级系统 son1,son2,son1 相对于 world,以及 son2 相对于 world 的关系是已知的,求 son1原点在 son2中的坐标,又已知在 son1中一点的坐标,要求求出该点在 son2 中的坐标

实现分析:

  1. 首先,需要发布 son1 相对于 world,以及 son2 相对于 world 的坐标消息
  2. 然后,需要订阅坐标发布消息,并取出订阅的消息,借助于 tf2 实现 son1 和 son2 的转换
  3. 最后,还要实现坐标点的转换

实现流程: C++ 与 Python 实现流程一致

  1. 新建功能包,添加依赖
  2. 创建坐标相对关系发布方(需要发布两个坐标相对关系)
  3. 创建坐标相对关系订阅方
  4. 执行

方案A:C++实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.发布方

为了方便,使用静态坐标变换发布(前面已经提到过静态坐标系数据发布的时候可以直接通过命令行发布,无需编程)

  1. <launch>
  2. <node pkg="tf2_ros" type="static_transform_publisher" name="son1" args="0.2 0.8 0.3 0 0 0 /world /son1" output="screen" />
  3. <node pkg="tf2_ros" type="static_transform_publisher" name="son2" args="0.5 0 0 0 0 0 /world /son2" output="screen" />
  4. </launch>

3.订阅方

  1. /*
  2. 需求:
  3. 现有坐标系统,父级坐标系统 world,下有两子级系统 son1son2
  4. son1 相对于 world,以及 son2 相对于 world 的关系是已知的,
  5. son1 son2中的坐标关系,又已知在 son1中一点的坐标,要求求出该点在 son2 中的坐标
  6. 实现流程:
  7. 1.包含头文件
  8. 2.初始化 ros 节点
  9. 3.创建 ros 句柄
  10. 4.创建 TF 订阅对象
  11. 5.解析订阅信息中获取 son1 坐标系原点在 son2 中的坐标
  12. 解析 son1 中的点相对于 son2 的坐标
  13. 6.spin
  14. */
  15. //1.包含头文件
  16. #include "ros/ros.h"
  17. #include "tf2_ros/transform_listener.h"
  18. #include "tf2/LinearMath/Quaternion.h"
  19. #include "tf2_geometry_msgs/tf2_geometry_msgs.h"
  20. #include "geometry_msgs/TransformStamped.h"
  21. #include "geometry_msgs/PointStamped.h"
  22. int main(int argc, char *argv[])
  23. {
  24. setlocale(LC_ALL,"");
  25. // 2.初始化 ros 节点
  26. ros::init(argc,argv,"sub_frames");
  27. // 3.创建 ros 句柄
  28. ros::NodeHandle nh;
  29. // 4.创建 TF 订阅对象
  30. tf2_ros::Buffer buffer; //实例化订阅对象
  31. tf2_ros::TransformListener listener(buffer);
  32. // 5.解析订阅信息中获取 son1 坐标系原点在 son2 中的坐标
  33. ros::Rate r(1);
  34. while (ros::ok())
  35. {
  36. try
  37. {
  38. // 解析 son1 中的点相对于 son2 的坐标 buffer.lookupTransform(“父坐标”,“子坐标”,缓冲区最新的tf数据)
  39. geometry_msgs::TransformStamped tfs = buffer.lookupTransform("son2","son1",ros::Time(0));
  40. ROS_INFO("Son1 相对于 Son2 的坐标关系:父坐标系ID=%s",tfs.header.frame_id.c_str());
  41. ROS_INFO("Son1 相对于 Son2 的坐标关系:子坐标系ID=%s",tfs.child_frame_id.c_str());
  42. ROS_INFO("Son1 相对于 Son2 的坐标关系:x=%.2f,y=%.2f,z=%.2f",
  43. tfs.transform.translation.x,
  44. tfs.transform.translation.y,
  45. tfs.transform.translation.z
  46. );
  47. // 坐标点解析(从这往后基本都一致了,关键在上面一步)
  48. geometry_msgs::PointStamped ps; //实例化坐标点对象
  49. ps.header.frame_id = "son1";
  50. ps.header.stamp = ros::Time::now();
  51. ps.point.x = 1.0;
  52. ps.point.y = 2.0;
  53. ps.point.z = 3.0;
  54. geometry_msgs::PointStamped psAtSon2;
  55. psAtSon2 = buffer.transform(ps,"son2");
  56. ROS_INFO("在 Son2 中的坐标:x=%.2f,y=%.2f,z=%.2f",
  57. psAtSon2.point.x,
  58. psAtSon2.point.y,
  59. psAtSon2.point.z
  60. );
  61. }
  62. catch(const std::exception& e)
  63. {
  64. // std::cerr << e.what() << '\n';
  65. ROS_INFO("异常信息:%s",e.what());
  66. }
  67. r.sleep();
  68. // 6.spin
  69. ros::spinOnce();
  70. }
  71. return 0;
  72. }

注意:

1,ros::Time::now()获取当前的时间,在时间“now”请求坐标系变换时,您应该等待几毫秒以获得该信息。

2,ros::Time(0)表示使用缓冲中最新的tf数据

关于ros::Time::now()和**ros::Time(0)的解释见:**https://blog.csdn.net/harrycomeon/article/details/116306149 配置文件此处略。

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,将输出换算后的结果。

方案B:Python实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.发布方

为了方便,使用静态坐标变换发布
  1. <launch>
  2. <node pkg="tf2_ros" type="static_transform_publisher" name="son1" args="0.2 0.8 0.3 0 0 0 /world /son1" output="screen" />
  3. <node pkg="tf2_ros" type="static_transform_publisher" name="son2" args="0.5 0 0 0 0 0 /world /son2" output="screen" />
  4. </launch>

3.订阅方

  1. #!/usr/bin/env python
  2. """
  3. 需求:
  4. 现有坐标系统,父级坐标系统 world,下有两子级系统 son1,son2,
  5. son1 相对于 world,以及 son2 相对于 world 的关系是已知的,
  6. 求 son1 与 son2中的坐标关系,又已知在 son1中一点的坐标,要求求出该点在 son2 中的坐标
  7. 实现流程:
  8. 1.导包
  9. 2.初始化 ROS 节点
  10. 3.创建 TF 订阅对象
  11. 4.调用 API 求出 son1 相对于 son2 的坐标关系
  12. 5.创建一依赖于 son1 的坐标点,调用 API 求出该点在 son2 中的坐标
  13. 6.spin
  14. """
  15. # 1.导包
  16. import rospy
  17. import tf2_ros
  18. from geometry_msgs.msg import TransformStamped
  19. from tf2_geometry_msgs import PointStamped
  20. if __name__ == "__main__":
  21. # 2.初始化 ROS 节点
  22. rospy.init_node("frames_sub_p")
  23. # 3.创建 TF 订阅对象
  24. buffer = tf2_ros.Buffer()
  25. listener = tf2_ros.TransformListener(buffer)
  26. rate = rospy.Rate(1)
  27. while not rospy.is_shutdown():
  28. try:
  29. # 4.调用 API 求出 son1 相对于 son2 的坐标关系
  30. #lookup_transform(self, target_frame, source_frame, time, timeout=rospy.Duration(0.0)):
  31. tfs = buffer.lookup_transform("son2","son1",rospy.Time(0))
  32. rospy.loginfo("son1 与 son2 相对关系:")
  33. rospy.loginfo("父级坐标系:%s",tfs.header.frame_id)
  34. rospy.loginfo("子级坐标系:%s",tfs.child_frame_id)
  35. rospy.loginfo("相对坐标:x=%.2f, y=%.2f, z=%.2f",
  36. tfs.transform.translation.x,
  37. tfs.transform.translation.y,
  38. tfs.transform.translation.z,
  39. )
  40. # 5.创建一依赖于 son1 的坐标点,调用 API 求出该点在 son2 中的坐标
  41. point_source = PointStamped() #实例化对象
  42. point_source.header.frame_id = "son1"
  43. point_source.header.stamp = rospy.Time.now()
  44. point_source.point.x = 1
  45. point_source.point.y = 1
  46. point_source.point.z = 1
  47. point_target = buffer.transform(point_source,"son2",rospy.Duration(0.5)) #每0.5秒进行转换
  48. rospy.loginfo("point_target 所属的坐标系:%s",point_target.header.frame_id)
  49. rospy.loginfo("坐标点相对于 son2 的坐标:(%.2f,%.2f,%.2f)",
  50. point_target.point.x,
  51. point_target.point.y,
  52. point_target.point.z
  53. )
  54. except Exception as e:
  55. rospy.logerr("错误提示:%s",e)
  56. rate.sleep()
  57. # 6.spin
  58. # rospy.spin()

注意:**.Timer(rospy.Duration(2),callback)#每2s触发一次**

权限设置以及配置文件此处略。

4.执行

可以使用命令行或launch文件的方式分别启动发布节点与订阅节点,如果程序无异常,将输出换算后的结果。

5、坐标系关系查看

5.1.5 坐标系关系查看

在机器人系统中,涉及的坐标系有多个,为了方便查看,ros 提供了专门的工具,可以用于生成显示坐标系关系的 pdf 文件,该文件包含树形结构的坐标系图谱。

准备

首先调用rospack find tf2_tools查看是否包含该功能包,如果没有,请使用如下命令安装:
  1. sudo apt install ros-noetic-tf2-tools

使用

生成 pdf 文件
启动坐标系广播程序之后,运行如下命令:
  1. rosrun tf2_tools view_frames.py
会产生类似于下面的日志信息:
  1. [INFO] [1592920556.827549]: Listening to tf data during 5 seconds...
  2. [INFO] [1592920561.841536]: Generating graph in frames.pdf file...
查看当前目录(默认的话是主目录)会生成一个 frames.pdf 文件

第五章 ROS常用组件. - 图11

查看文件
可以直接进入目录打开文件,或者调用命令查看文件:evince frames.pdf 内如如图所示:
第五章 ROS常用组件. - 图12 第五章 ROS常用组件. - 图13 ## 6、TF坐标变空换实操 ** 需求描述:** 程序启动之初: 产生两只乌龟,中间的乌龟(A) 和 左下乌龟(B), B 会自动运行至A的位置,并且键盘控制时,只是控制 A 的运动,但是 B 可以跟随 A 运行

结果演示:

第五章 ROS常用组件. - 图14

实现分析:

乌龟跟随实现的核心,是乌龟A和B都要发布相对世界坐标系的坐标信息,然后,订阅到该信息需要转换获取A相对于B坐标系的信息,最后,再生成速度信息,并控制B运动。
  1. 启动乌龟显示节点
  2. 在乌龟显示窗体中生成一只新的乌龟(需要使用服务)
  3. 编写两只乌龟发布坐标信息的节点
  4. 编写订阅节点订阅坐标信息并生成新的相对关系生成速度信息

实现流程:C++ 与 Python 实现流程一致

  1. 新建功能包,添加依赖
  2. 编写服务客户端,用于生成一只新的乌龟
  3. 编写发布方,发布两只乌龟的坐标信息
  4. 编写订阅方,订阅两只乌龟信息,生成速度信息并发布
  5. 运行

准备工作:

1.了解如何创建第二只乌龟,且不受键盘控制

创建第二只乌龟需要使用rosservice,服务对象名称使用的是 spawn

  1. rosservice call /spawn "x: 1.0
  2. y: 1.0
  3. theta: 1.0
  4. name: 'turtle_flow'"
  5. name: "turtle_flow"
键盘是无法控制第二只乌龟运动的,因为使用的话题: /第二只乌龟名称/cmd_vel,对应的要控制乌龟运动必须发布对应的话题消息 2.了解如何获取两只乌龟的坐标

是通过**话题 /乌龟名称/pose **来获取的

  1. x: 1.0 //x坐标
  2. y: 1.0 //y坐标
  3. theta: -1.21437060833 //角度
  4. linear_velocity: 0.0 //线速度
  5. angular_velocity: 1.0 //角速度

方案A:C++实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.服务客户端请求(生成乌龟)

第五章 ROS常用组件. - 图15

第五章 ROS常用组件. - 图16注意:

rossrv与rosservice的区别:

rossrv是用于显示有关ROS服务类型的信息的命令行工具,与 rosmsg 使用语法高度雷同。**在显示)**

  1. rosmsg list 列出所有消息

rosservice包含用于列出查询ROSServices的rosservice命令行工具。调用部分服务时,如果对相关工作空间没有配置 path,需要进入工作空间调用 source ./devel/setup.bash

  1. rosservice info 打印有关服务的信息
  1. /*
  2. 创建第二只小乌龟
  3. */
  4. #include "ros/ros.h"
  5. #include "turtlesim/Spawn.h"
  6. int main(int argc, char *argv[])
  7. {
  8. setlocale(LC_ALL,"");
  9. //执行初始化
  10. ros::init(argc,argv,"create_turtle");
  11. //创建节点
  12. ros::NodeHandle nh;
  13. //创建服务客户端
  14. ros::ServiceClient client = nh.serviceClient<turtlesim::Spawn>("/spawn");
  15. ros::service::waitForService("/spawn");
  16. turtlesim::Spawn spawn;
  17. spawn.request.name = "turtle2";
  18. spawn.request.x = 1.0;
  19. spawn.request.y = 2.0;
  20. spawn.request.theta = 3.12415926;
  21. bool flag = client.call(spawn);
  22. if (flag)
  23. {
  24. ROS_INFO("乌龟%s创建成功!",spawn.response.name.c_str());
  25. }
  26. else
  27. {
  28. ROS_INFO("乌龟2创建失败!");
  29. }
  30. ros::spin();
  31. return 0;
  32. }

配置文件此处略。

3.发布方(发布两只乌龟的坐标信息)

可以订阅乌龟的位姿信息,然后再转换成坐标信息,两只乌龟的实现逻辑相同,只是订阅的话题名称,生成的坐标信息等稍有差异,可以将差异部分通过参数传入:
  • 该节点**需要启动两次(因为是两只乌龟)**
  • 每次启动时都需要传入乌龟节点名称(第一次是 turtle1, 第二次是 turtle2)
  1. /*
  2. 该文件实现:需要订阅 turtle1 turtle2 pose,然后广播相对 world 的坐标系信息
  3. 注意: 订阅的两只 turtle,除了命名空间(turtle1 turtle2)不同外,
  4. 其他的话题名称和实现逻辑都是一样的!!!!!
  5. 所以我们可以将所需的命名空间通过 args 动态传入
  6. 实现流程:
  7. 1.包含头文件
  8. 2.初始化 ros 节点
  9. 3.解析传入的命名空间
  10. 4.创建 ros 句柄
  11. 5.创建订阅对象
  12. 6.回调函数处理订阅的 pose 信息
  13. 6-1.创建 TF 广播器
  14. 6-2. pose 信息转换成 TransFormStamped
  15. 6-3.发布
  16. 7.spin
  17. */
  18. //1.包含头文件
  19. #include "ros/ros.h"
  20. #include "turtlesim/Pose.h"
  21. #include "tf2_ros/transform_broadcaster.h"
  22. #include "tf2/LinearMath/Quaternion.h"
  23. #include "geometry_msgs/TransformStamped.h"
  24. //保存乌龟名称
  25. std::string turtle_name; //实例化这个字符串对象
  26. void doPose(const turtlesim::Pose::ConstPtr& pose){
  27. // 6-1.创建 TF 广播器 ---------------------------------------- 注意 static
  28. static tf2_ros::TransformBroadcaster broadcaster;
  29. // 6-2. pose 信息转换成 TransFormStamped
  30. geometry_msgs::TransformStamped tfs;
  31. tfs.header.frame_id = "world";
  32. tfs.header.stamp = ros::Time::now();
  33. tfs.child_frame_id = turtle_name;
  34. tfs.transform.translation.x = pose->x;
  35. tfs.transform.translation.y = pose->y;
  36. tfs.transform.translation.z = 0.0;
  37. tf2::Quaternion qtn;
  38. qtn.setRPY(0,0,pose->theta);
  39. tfs.transform.rotation.x = qtn.getX();
  40. tfs.transform.rotation.y = qtn.getY();
  41. tfs.transform.rotation.z = qtn.getZ();
  42. tfs.transform.rotation.w = qtn.getW();
  43. // 6-3.发布
  44. broadcaster.sendTransform(tfs);
  45. }
  46. int main(int argc, char *argv[])
  47. {
  48. setlocale(LC_ALL,"");
  49. // 2.初始化 ros 节点
  50. ros::init(argc,argv,"pub_tf");
  51. // 3.解析传入的命名空间 比如等于2
  52. if (argc != 2)
  53. {
  54. ROS_ERROR("请传入正确的参数");
  55. } else {
  56. turtle_name = argv[1]; //把名字提取出来
  57. ROS_INFO("乌龟 %s 坐标发送启动",turtle_name.c_str());
  58. }
  59. // 4.创建 ros 句柄
  60. ros::NodeHandle nh;
  61. // 5.创建订阅对象
  62. ros::Subscriber sub = nh.subscribe<turtlesim::Pose>(turtle_name + "/pose",1000,doPose);
  63. // 6.回调函数处理订阅的 pose 信息
  64. // 6-1.创建 TF 广播器
  65. // 6-2. pose 信息转换成 TransFormStamped
  66. // 6-3.发布
  67. // 7.spin
  68. ros::spin();
  69. return 0;
  70. }

配置文件此处略。

4.订阅方(解析坐标信息并生成速度信息)

  1. /*
  2. 订阅 turtle1 turtle2 TF 广播信息,查找并转换时间最近的 TF 信息
  3. turtle1 转换成相对 turtle2 的坐标,在计算线速度和角速度并发布
  4. 实现流程:
  5. 1.包含头文件
  6. 2.初始化 ros 节点
  7. 3.创建 ros 句柄
  8. 4.创建 TF 订阅对象
  9. 5.处理订阅到的 TF
  10. 6.spin
  11. */
  12. //1.包含头文件
  13. #include "ros/ros.h"
  14. #include "tf2_ros/transform_listener.h"
  15. #include "geometry_msgs/TransformStamped.h"
  16. #include "geometry_msgs/Twist.h"
  17. int main(int argc, char *argv[])
  18. {
  19. setlocale(LC_ALL,"");
  20. // 2.初始化 ros 节点
  21. ros::init(argc,argv,"sub_TF");
  22. // 3.创建 ros 句柄
  23. ros::NodeHandle nh;
  24. // 4.创建 TF 订阅对象
  25. tf2_ros::Buffer buffer;
  26. tf2_ros::TransformListener listener(buffer);
  27. // 5.处理订阅到的 TF
  28. // 需要创建发布 /turtle2/cmd_vel publisher 对象
  29. ros::Publisher pub = nh.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel",1000);
  30. ros::Rate rate(10);
  31. while (ros::ok())
  32. {
  33. try
  34. {
  35. //5-1.先获取 turtle1 相对 turtle2 的坐标信息
  36. geometry_msgs::TransformStamped tfs = buffer.lookupTransform("turtle2","turtle1",ros::Time(0));
  37. //5-2.根据坐标信息生成速度信息 -- geometry_msgs/Twist.h
  38. geometry_msgs::Twist twist;
  39. twist.linear.x = 0.5 * sqrt(pow(tfs.transform.translation.x,2) + pow(tfs.transform.translation.y,2));
  40. twist.angular.z = 4 * atan2(tfs.transform.translation.y,tfs.transform.translation.x);
  41. //5-3.发布速度信息 -- 需要提前创建 publish 对象 前面已经创建了
  42. pub.publish(twist);
  43. }
  44. catch(const std::exception& e)
  45. {
  46. // std::cerr << e.what() << '\n';
  47. ROS_INFO("错误提示:%s",e.what());
  48. }
  49. rate.sleep();
  50. // 6.spin
  51. ros::spinOnce();
  52. }
  53. return 0;
  54. }

配置文件此处略。

5.运行

使用 launch 文件组织需要运行的节点,内容示例如下:
  1. <!--
  2. tf2 实现小乌龟跟随案例
  3. -->
  4. <launch>
  5. <!-- 启动乌龟节点与键盘控制节点 -->
  6. <node pkg="turtlesim" type="turtlesim_node" name="turtle1" output="screen" />
  7. <node pkg="turtlesim" type="turtle_teleop_key" name="key_control" output="screen"/>
  8. <!-- 启动创建第二只乌龟的节点 -->
  9. <node pkg="demo_tf2_test" type="Test01_Create_Turtle2" name="turtle2" output="screen" />
  10. <!-- 启动两个坐标发布节点 -->
  11. <node pkg="demo_tf2_test" type="Test02_TF2_Caster" name="caster1" output="screen" args="turtle1" />
  12. <node pkg="demo_tf2_test" type="Test02_TF2_Caster" name="caster2" output="screen" args="turtle2" />
  13. <!-- 启动坐标转换节点 -->
  14. <node pkg="demo_tf2_test" type="Test03_TF2_Listener" name="listener" output="screen" />
  15. </launch>
方案B:Python实现

1.创建功能包

创建项目功能包依赖于 tf2、tf2_ros、tf2_geometry_msgs、roscpp rospy std_msgs geometry_msgs、turtlesim

2.服务客户端(生成乌龟)

  1. #! /usr/bin/env python
  2. """
  3. 调用 service 服务在窗体指定位置生成一只乌龟
  4. 流程:
  5. 1.导包
  6. 2.初始化 ros 节点
  7. 3.创建服务客户端
  8. 4.等待服务启动
  9. 5.创建请求数据
  10. 6.发送请求并处理响应
  11. """
  12. #1.导包
  13. import rospy
  14. from turtlesim.srv import Spawn, SpawnRequest, SpawnResponse
  15. if __name__ == "__main__":
  16. # 2.初始化 ros 节点
  17. rospy.init_node("turtle_spawn_p")
  18. # 3.创建服务客户端
  19. client = rospy.ServiceProxy("/spawn",Spawn)
  20. # 4.等待服务启动
  21. client.wait_for_service()
  22. # 5.创建请求数据
  23. req = SpawnRequest()
  24. req.x = 1.0
  25. req.y = 1.0
  26. req.theta = 3.14
  27. req.name = "turtle2"
  28. # 6.发送请求并处理响应
  29. try:
  30. response = client.call(req)
  31. rospy.loginfo("乌龟创建成功,名字是:%s",response.name)
  32. except Exception as e:
  33. rospy.loginfo("服务调用失败....")
权限设置以及配置文件此处略。

3.发布方(发布两只乌龟的坐标信息) (订阅后广播

  1. #! /usr/bin/env python
  2. """
  3. 该文件实现:需要订阅 turtle1 和 turtle2 的 pose,然后广播相对 world 的坐标系信息
  4. 注意: 订阅的两只 turtle,除了命名空间(turtle1 和 turtle2)不同外,
  5. 其他的话题名称和实现逻辑都是一样的,
  6. 所以我们可以将所需的命名空间通过 args 动态传入
  7. 实现流程:
  8. 1.导包
  9. 2.初始化 ros 节点
  10. 3.解析传入的命名空间
  11. 4.创建订阅对象
  12. 5.回调函数处理订阅的 pose 信息
  13. 5-1.创建 TF 广播器
  14. 5-2.将 pose 信息转换成 TransFormStamped
  15. 5-3.发布
  16. 6.spin
  17. """
  18. # 1.导包
  19. import rospy
  20. import sys
  21. from turtlesim.msg import Pose
  22. from geometry_msgs.msg import TransformStamped
  23. import tf2_ros
  24. import tf_conversions
  25. turtle_name = ""
  26. def doPose(pose):
  27. # rospy.loginfo("x = %.2f",pose.x)
  28. #1.创建坐标系广播器
  29. broadcaster = tf2_ros.TransformBroadcaster()
  30. #2.将 pose 信息转换成 TransFormStamped
  31. tfs = TransformStamped()
  32. tfs.header.frame_id = "world"
  33. tfs.header.stamp = rospy.Time.now()
  34. tfs.child_frame_id = turtle_name
  35. tfs.transform.translation.x = pose.x
  36. tfs.transform.translation.y = pose.y
  37. tfs.transform.translation.z = 0.0
  38. qtn = tf_conversions.transformations.quaternion_from_euler(0, 0, pose.theta)
  39. tfs.transform.rotation.x = qtn[0]
  40. tfs.transform.rotation.y = qtn[1]
  41. tfs.transform.rotation.z = qtn[2]
  42. tfs.transform.rotation.w = qtn[3]
  43. #3.广播器发布 tfs
  44. broadcaster.sendTransform(tfs)
  45. if __name__ == "__main__":
  46. # 2.初始化 ros 节点
  47. rospy.init_node("sub_tfs_p")
  48. # 3.解析传入的命名空间
  49. rospy.loginfo("-------------------------------%d",len(sys.argv))
  50. if len(sys.argv) < 2:
  51. rospy.loginfo("请传入参数:乌龟的命名空间")
  52. else:
  53. turtle_name = sys.argv[1]
  54. rospy.loginfo("///////////////////乌龟:%s",turtle_name)
  55. rospy.Subscriber(turtle_name + "/pose",Pose,doPose)
  56. # 4.创建订阅对象
  57. # 5.回调函数处理订阅的 pose 信息
  58. # 5-1.创建 TF 广播器
  59. # 5-2.将 pose 信息转换成 TransFormStamped
  60. # 5-3.发布
  61. # 6.spin
  62. rospy.spin()
权限设置以及配置文件此处略。

4.订阅方(解析坐标信息并生成速度信息)

  1. #! /usr/bin/env python
  2. """
  3. 订阅 turtle1 和 turtle2 的 TF 广播信息,查找并转换时间最近的 TF 信息
  4. 将 turtle1 转换成相对 turtle2 的坐标,在计算线速度和角速度并发布
  5. 实现流程:
  6. 1.导包
  7. 2.初始化 ros 节点
  8. 3.创建 TF 订阅对象
  9. 4.处理订阅到的 TF
  10. 4-1.查找坐标系的相对关系
  11. 4-2.生成速度信息,然后发布
  12. """
  13. # 1.导包
  14. import rospy
  15. import tf2_ros
  16. from geometry_msgs.msg import TransformStamped, Twist
  17. import math
  18. if __name__ == "__main__":
  19. # 2.初始化 ros 节点
  20. rospy.init_node("sub_tfs_p")
  21. # 3.创建 TF 订阅对象
  22. buffer = tf2_ros.Buffer()
  23. listener = tf2_ros.TransformListener(buffer)
  24. # 4.处理订阅到的 TF
  25. rate = rospy.Rate(10)
  26. # 创建速度发布对象
  27. pub = rospy.Publisher("/turtle2/cmd_vel",Twist,queue_size=1000)
  28. while not rospy.is_shutdown():
  29. rate.sleep()
  30. try:
  31. #def lookup_transform(self, target_frame, source_frame, time, timeout=rospy.Duration(0.0)):
  32. trans = buffer.lookup_transform("turtle2","turtle1",rospy.Time(0))
  33. # rospy.loginfo("相对坐标:(%.2f,%.2f,%.2f)",
  34. # trans.transform.translation.x,
  35. # trans.transform.translation.y,
  36. # trans.transform.translation.z
  37. # )
  38. # 根据转变后的坐标计算出速度和角速度信息
  39. twist = Twist()
  40. # 间距 = x^2 + y^2 然后开方
  41. twist.linear.x = 0.5 * math.sqrt(math.pow(trans.transform.translation.x,2) + math.pow(trans.transform.translation.y,2))
  42. twist.angular.z = 4 * math.atan2(trans.transform.translation.y, trans.transform.translation.x)
  43. pub.publish(twist)
  44. except Exception as e:
  45. rospy.logwarn("警告:%s",e)
权限设置以及配置文件此处略。

5.运行

使用 launch 文件组织需要运行的节点,内容示例如下:
  1. <launch>
  2. <node pkg="turtlesim" type="turtlesim_node" name="turtle1" output="screen" />
  3. <node pkg="turtlesim" type="turtle_teleop_key" name="key_control" output="screen"/>
  4. <node pkg="demo06_test_flow_p" type="test01_turtle_spawn_p.py" name="turtle_spawn" output="screen"/>
  5. <node pkg="demo06_test_flow_p" type="test02_turtle_tf_pub_p.py" name="tf_pub1" args="turtle1" output="screen"/>
  6. <node pkg="demo06_test_flow_p" type="test02_turtle_tf_pub_p.py" name="tf_pub2" args="turtle2" output="screen"/>
  7. <node pkg="demo06_test_flow_p" type="test03_turtle_tf_sub_p.py" name="tf_sub" output="screen"/>
  8. </launch>

7、TF2与TF

1.TF2与TF比较_简介

  • TF2已经替换了TF,TF2是TF的超集,建议学习 TF2 而非 TF
  • TF2 功能包的增强了内聚性,TF 与 TF2 所依赖的功能包是不同的,TF 对应的是tf包,TF2 对应的是tf2tf2_ros包,在 TF2 中不同类型的 API 实现做了分包处理。
  • TF2 实现效率更高,比如在:TF2 的静态坐标实现、TF2 坐标变换监听器中的 Buffer 实现等

2.TF2与TF比较_静态坐标变换演示

接下来,我们通过静态坐标变换来演示TF2的实现效率。
2.1启动 TF2 与 TF 两个版本的静态坐标变换

TF2 版静态坐标变换:

  1. rosrun tf2_ros static_transform_publisher 0 0 0 0 0 0 /base_link /laser

TF 版静态坐标变换:

  1. rosrun tf static_transform_publisher 0 0 0 0 0 0 /base_link /laser 100

会发现,TF 版本的启动中最后多一个参数,该参数是指定发布频率

2.2运行结果比对
使用rostopic查看话题,包含/tf/tf_static, 前者是 TF 发布的话题,后者是 TF2 发布的话题,分别调用命令打印二者的话题消息
TF TF2
/tf /tf_static
循环输出坐标系信息 坐标系信息只有一次
rostopic echo /tf: 当前会循环输出坐标系信息

rostopic echo /tf_static**: 坐标系信息只有一次**

2.3结论
如果是静态坐标转换,那么不同坐标系之间的相对状态是固定的,既然是固定的,那么没有必要重复发布坐标系的转换消息,很显然的,tf2 实现较之于 tf 更为高效

8、小结

坐标变换在机器人系统中是一个极其重要的组成模块,在 ROS 中 TF2 组件是专门用于实现坐标变换的,TF2 实现具体内容又主要介绍了如下几部分:

1.静态坐标变换广播器,可以编码方式或调用内置功能包来实现(建议后者),适用于相对固定的坐标系关系 2.动态坐标变换广播器,**以编码的方式广播坐标系之间的相对关系**,适用于易变的坐标系关系 3.坐标变换监听器,用于监听广播器广播的坐标系消息,可以实现不同坐标系之间或同一点在不同坐标系之间的变换 4.机器人系统中的坐标系关系是较为复杂的,还可以通过 tf2_tools 工具包来生成 ros 中的坐标系关系图 5.当前 TF2 已经替换了 TF,官网建议直接学习 TF2,并且 TF 与 TF2 的使用流程与实现 API 比较类似,只要有任意一方的使用经验,另一方也可以做到触类旁通

5.2 rosbag

机器人传感器获取到的信息,有时我们可能需要时时处理,有时可能只是**采集数据,事后分析**,比如:

机器人导航实现中,可能需要绘制导航所需的全局地图,地图绘制实现,有两种方式, 方式1:可以控制机器人运动,将机器人传感器感知到的数据时时处理,生成地图信息。 方式2:同样是控制机器人运动,将机器人传感器感知到的数据留存,事后,再重新读取数据,生成地图信息。两种方式比较,显然方式2使用上更为灵活方便 在ROS中关于数据的留存以及读取实现,提供了专门的工具: rosbag。 —- 概念 是用于录制和回放 ROS 主题的一个工具集。

作用

实现了数据的复用,方便调试、测试

本质

rosbag本质也是ros的节点,当录制时,rosbag是一个订阅节点,可以订阅话题消息并将订阅到的数据写入磁盘文件当重放时,rosbag是一个发布节点,可以读取磁盘文件,发布文件中的话题消息

1、rosbag使用命令行


需求:

ROS 内置的乌龟案例并操作,操作过程中使用 rosbag 录制,录制结束后,实现重放

实现:

1.准备 创建目录保存录制的文件
  1. mkdir ./xxx cd xxx
2.开始录制
  1. rosbag record -a -O 目标文件
操作小乌龟一段时间,结束录制使用 ctrl + c,在创建的目录中会生成bag文件。 3.查看文件
  1. rosbag info 文件名
4.回放文件 play
  1. rosbag play 文件名
重启乌龟节点,会发现,乌龟按照录制时的轨迹运动。

2、rosbag使用编码

命令实现不够灵活,可以使用编码的方式,增强录制与回放的灵活性,本节将通过简单的读写实现演示rosbag的编码实现。


方案A:C++实现

1.写 bag

  1. #include "ros/ros.h"
  2. #include "rosbag/bag.h" //新加
  3. #include "std_msgs/String.h"
  4. int main(int argc, char *argv[])
  5. {
  6. ros::init(argc,argv,"bag_write");
  7. ros::NodeHandle nh;
  8. //创建bag对象 实例化
  9. rosbag::Bag bag;
  10. //打开
  11. bag.open("/home/rosdemo/demo/test.bag",rosbag::BagMode::Write);
  12. //写
  13. std_msgs::String msg;
  14. msg.data = "hello world";
  15. bag.write("/chatter",ros::Time::now(),msg);
  16. bag.write("/chatter",ros::Time::now(),msg);
  17. bag.write("/chatter",ros::Time::now(),msg);
  18. bag.write("/chatter",ros::Time::now(),msg);
  19. //关闭
  20. bag.close();
  21. return 0;
  22. }

2.读bag

  1. /*
  2. 读取 bag 文件:
  3. */
  4. #include "ros/ros.h"
  5. #include "rosbag/bag.h"
  6. #include "rosbag/view.h"
  7. #include "std_msgs/String.h"
  8. #include "std_msgs/Int32.h"
  9. int main(int argc, char *argv[])
  10. {
  11. setlocale(LC_ALL,"");
  12. ros::init(argc,argv,"bag_read");
  13. ros::NodeHandle nh;
  14. //创建 bag 对象
  15. rosbag::Bag bag;
  16. //打开 bag 文件
  17. bag.open("/home/rosdemo/demo/test.bag",rosbag::BagMode::Read);
  18. //读数据
  19. for (rosbag::MessageInstance const m : rosbag::View(bag))
  20. {
  21. std_msgs::String::ConstPtr p = m.instantiate<std_msgs::String>();
  22. if(p != nullptr){
  23. ROS_INFO("读取的数据:%s",p->data.c_str());
  24. }
  25. }
  26. //关闭文件流
  27. bag.close();
  28. return 0;
  29. }
方案B:Python实现(还是python实现起来简单啊!!!)

1.写 bag

  1. #! /usr/bin/env python
  2. import rospy
  3. import rosbag
  4. from std_msgs.msg import String
  5. if __name__ == "__main__":
  6. #初始化节点
  7. rospy.init_node("w_bag_p")
  8. # 创建 rosbag 对象
  9. bag = rosbag.Bag("/home/rosdemo/demo/test.bag",'w')
  10. # 写数据
  11. s = String()
  12. s.data= "hahahaha"
  13. bag.write("chatter",s)
  14. bag.write("chatter",s)
  15. bag.write("chatter",s)
  16. # 关闭流
  17. bag.close()

2.读bag

  1. #! /usr/bin/env python
  2. import rospy
  3. import rosbag
  4. from std_msgs.msg import String
  5. if __name__ == "__main__":
  6. #初始化节点
  7. rospy.init_node("w_bag_p")
  8. # 创建 rosbag 对象
  9. bag = rosbag.Bag("/home/rosdemo/demo/test.bag",'r')
  10. # 读数据
  11. bagMessage = bag.read_messages("chatter")
  12. for topic,msg,t in bagMessage:
  13. rospy.loginfo("%s,%s,%s",topic,msg,t)
  14. # 关闭流
  15. bag.close()

3rqt工具箱

之前,在 ROS 中使用了一些实用的工具,比如: **ros_bag 用于录制与回放tf2_tools 可以生成 TF 树 ….. 这些工具大大提高了开发的便利性,但是也存在一些问题: 这些工具的启动和使用过程中涉及到一些命令操作,应用起来不够方便,在ROS中,提供了rqt工具箱在调用工具时以图形化操作代替了命令操作**,应用更便利,提高了操作效率,优化了用户体验。


概念

ROS基于 QT 框架,针对机器人开发提供了一系列可视化的工具,这些工具的集合就是rqt

作用

可以方便的实现 ROS 可视化调试,并且在同一窗口中打开多个部件,提高开发效率,优化用户体验。

组成

rqt 工具箱组成有三大部分
  • rqt——核心实现,开发人员无需关注
  • rqt_common_plugins——rqt 中常用的工具套件 plugins插件的意思
  • rqt_robot_plugins——运行中和机器人交互的插件(比如: rviz)

1、rq安装启动与基本使用

1.安装

  • 一般只要你安装的是desktop-full版本就会自带工具箱
  • 如果需要安装可以以如下方式安装
  1. $ sudo apt-get install ros-noetic-rqt
  2. $ sudo apt-get install ros-noetic-rqt-common-plugins

2.启动

rqt的启动方式有两种:
  • 方式1:**rqt**
  • 方式2:rosrun rqt_gui rqt_gui

3.基本使用

启动 rqt 之后,可以通过 plugins 添加所需的插件第五章 ROS常用组件. - 图17 ## 2、rqt第用件 rat graph ## 简介:可视化显示计算图 启动:可以在 rqt 的 plugins 中添加,或者使用rqt_graph启动

第五章 ROS常用组件. - 图18

3、rqt常用件 rgt console

简介:rqt_console 是 ROS 中用于显示和过滤日志的图形化插件

准备:编写 Node 节点输出各个级别的日志信息

  1. /*
  2. ROS 节点:输出各种级别的日志信息
  3. */
  4. #include "ros/ros.h"
  5. int main(int argc, char *argv[])
  6. {
  7. ros::init(argc,argv,"log_demo");
  8. ros::NodeHandle nh;
  9. ros::Rate r(0.3);
  10. while (ros::ok())
  11. {
  12. ROS_DEBUG("Debug message d");
  13. ROS_INFO("Info message oooooooooooooo");
  14. ROS_WARN("Warn message wwwww");
  15. ROS_ERROR("Erroe message EEEEEEEEEEEEEEEEEEEE");
  16. ROS_FATAL("Fatal message FFFFFFFFFFFFFFFFFFFFFFFFFFFFF");
  17. r.sleep();
  18. }
  19. return 0;
  20. }

启动:

可以在 rqt 的 plugins 中添加,或者使用rqt_console启动第五章 ROS常用组件. - 图19 ## 4、rqt常用件 rqt plot 简介:图形绘制插件,可以**以 2D 绘图的方式绘制发布在 topic 上的数据 准备:**启动 turtlesim 乌龟节点与键盘控制节点,通过 rqt_plot 获取乌龟位姿

启动:可以在 rqt 的 plugins 中添加,或者使用rqt_plot启动第五章 ROS常用组件. - 图20

5、rqt常用件 rqt bag

简介:录制和重放 bag 文件的图形化插件

准备:启动 turtlesim 乌龟节点与键盘控制节点

启动:可以在 rqt 的 plugins 中添加,或者使用rqt_bag启动

录制:第五章 ROS常用组件. - 图21

重放:第五章 ROS常用组件. - 图22

5.4本章小结

本章主要介绍了ROS中的常用组件,内容如下:
  • TF坐标变换(重点)
  • rosbag 用于ros话题的录制与回放
  • rqt工具箱,图形化方式调用组件,提高操作效率以及易用性(集成的小工具箱)
其中 TF坐标变换是重点,也是难点,需要大家熟练掌握坐标变换的应用场景以及代码实现。下一章开始将介绍机器人系统仿真,我们将在仿真环境下,创建机器人、控制机器人运动、搭建仿真环境,并以机器人的视角去感知世界。