• 第 5 章 TF坐标变换(自学二刷笔记)


    重要参考:

    课程链接:https://www.bilibili.com/video/BV1Ci4y1L7ZZ

    讲义链接:Introduction · Autolabor-ROS机器人入门课程《ROS理论与实践》零基础教程

    5.1.6 TF坐标变换实操

    需求描述:

    程序启动之初:产生两只乌龟,中间的乌龟(A) 和 左下乌龟(B),B 会自动运行至A的位置,并且键盘控制时,只是控制 A 的运动,但是 B 可以跟随 A 运行

    结果演示:

    实现分析:

    乌龟跟随实现的核心,是乌龟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.服务客户端(生成乌龟)
    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("/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.解析传入的命名空间
    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(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("/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. <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="demo_tf2_test" type="Test01_Create_Turtle2" name="turtle2" output="screen" />
    5. <node pkg="demo_tf2_test" type="Test02_TF2_Caster" name="caster1" output="screen" args="turtle1" />
    6. <node pkg="demo_tf2_test" type="Test02_TF2_Caster" name="caster2" output="screen" args="turtle2" />
    7. <node pkg="demo_tf2_test" type="Test03_TF2_Listener" name="listener" output="screen" />
    8. 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>
  • 相关阅读:
    [补题记录] Atcoder Beginner Contest 297(F)
    【信管1.12】新技术(一)物联网与云计算
    springcloud
    逆向-还原代码之运算符++ (Interl 32)
    LeetCode146.LRU缓存
    图学习——06.metapath2vec
    Java项目:SSM场地预订管理系统
    人工智能科学计算库—Numpy教程
    echarts常用参数详解汇总(饼图,柱形图,折线图)持续更新中
    [Docker]八.Docker 容器跨主机通讯
  • 原文地址:https://blog.csdn.net/Galaxy_1229/article/details/136669724