当前位置:   article > 正文

ROS:TF笔记_tf_buffer.transform(

tf_buffer.transform(

TF坐标变换,实现不同类型的坐标系之间的转换

tf2常用功能包有:

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

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

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

TF中常用的msg:geometry_msgs/TransformStampedgeometry_msgs/PointStamped

geometry_msgs/TransformStamped

  1. std_msgs/Header header
  2. uint32 seq
  3. time stamp
  4. string frame_id
  5. string child_frame_id
  6. geometry_msgs/Transform transform
  7. geometry_msgs/Vector3 translation
  8. float64 x
  9. float64 y
  10. float64 z
  11. geometry_msgs/Quaternion rotation
  12. float64 x
  13. float64 y
  14. float64 z
  15. float64 w

geometry_msgs/PointStamped

  1. std_msgs/Header header
  2. uint32 seq
  3. time stamp
  4. string frame_id
  5. geometry_msgs/Point point
  6. float64 x
  7. float64 y
  8. float64 z

静态坐标变换

发布坐标转换

  1. #include "ros/ros.h"
  2. #include "tf2_ros/static_transform_broadcaster.h"
  3. #include "geometry_msgs/TransformStamped.h"
  4. #include "tf/LinearMath/Quaternion.h"
  5. int main(int argc, char *argv[])
  6. {
  7. setlocale(LC_ALL,"");
  8. ros::init(argc, argv, "static_pub");
  9. ros::NodeHandle n;
  10. tf2_ros::StaticTransformBroadcaster pub; //创建发布对象
  11. geometry_msgs::TransformStamped tfs; //创建坐标转换消息
  12. tfs.header.stamp = ros::Time::now();
  13. tfs.header.frame_id = "base_link";
  14. tfs.child_frame_id = "laser";
  15. tfs.transform.translation.x = 0.2;
  16. tfs.transform.translation.y = 0;
  17. tfs.transform.translation.z = 0.5;
  18. tf::Quaternion qtn; //通过欧拉角设置四元数 包含"tf/LinearMath/Quaternion.h"
  19. qtn.setRPY(0, 0, 0);
  20. tfs.transform.rotation.x = qtn.getX();
  21. tfs.transform.rotation.y = qtn.getY();
  22. tfs.transform.rotation.z = qtn.getZ();
  23. tfs.transform.rotation.w = qtn.getW();
  24. pub.sendTransform(tfs);
  25. ros::spin();
  26. return 0;
  27. }
  1. #include "ros/ros.h"
  2. #include "tf2_ros/transform_listener.h"
  3. #include "tf2_ros/buffer.h"
  4. #include "geometry_msgs/PointStamped.h"
  5. #include "tf2_geometry_msgs/tf2_geometry_msgs.h"
  6. int main(int argc, char *argv[])
  7. {
  8. setlocale(LC_ALL, "");
  9. ros::init(argc, argv, "static_sub");
  10. ros::NodeHandle n;
  11. tf2_ros::Buffer buffer; //创建buffer对象,存储坐标转换
  12. tf2_ros::TransformListener listener(buffer); //创建接收对象,传入buffer
  13. geometry_msgs::PointStamped p; //创建点
  14. p.header.frame_id = "laser";
  15. p.header.stamp = ros::Time::now();
  16. p.point.x = 2.0;
  17. p.point.y = 3.0;
  18. p.point.z = 5.0;
  19. // ros::Duration(2).sleep(); //方案1:休眠解决"base_link" passed to lookupTransform argument target_frame does not exist.
  20. ros::Rate rate(10);
  21. while (ros::ok())
  22. {
  23. geometry_msgs::PointStamped p_out;
  24. try //方案2:try解决
  25. {
  26. p_out = buffer.transform(p, "base_link"); //需包含"tf2_geometry_msgs/tf2_geometry_msgs.h"
  27. ROS_INFO("\033[1;32m%.2f %.2f %.2f 参考坐标系:%s\033[0m", p_out.point.x,
  28. p_out.point.y, p_out.point.z, p_out.header.frame_id.c_str());
  29. }
  30. catch(const std::exception& e)
  31. {
  32. // std::cerr << e.what() << '\n';
  33. ROS_INFO("\033[31m %s \033[0m", e.what());
  34. }
  35. rate.sleep();
  36. ros::spinOnce();
  37. }
  38. return 0;
  39. }

直接调用内置功能包实现

rosrun tf2_ros static_transform_publisher x偏移量 y偏移量 z偏移量 z偏航角度 y俯仰角度 x翻滚角度 父级坐标系 子级坐标系

动态坐标变换

订阅小乌龟位姿发布to world的坐标变换在rviz中显示

  1. #include "ros/ros.h"
  2. #include "turtlesim/Pose.h"
  3. #include "tf2_ros/transform_broadcaster.h"
  4. #include "geometry_msgs/TransformStamped.h"
  5. #include "tf2/LinearMath/Quaternion.h"
  6. void doPose(const turtlesim::Pose::ConstPtr& msg){
  7. static tf2_ros::TransformBroadcaster pub;
  8. geometry_msgs::TransformStamped ts;
  9. ts.header.frame_id = "world";
  10. ts.header.stamp = ros::Time::now();
  11. ts.child_frame_id = "turtle1";
  12. ts.transform.translation.x = msg->x;
  13. ts.transform.translation.y = msg->y;
  14. ts.transform.translation.z = 0;
  15. tf2::Quaternion qtn;
  16. qtn.setRPY(0, 0, msg->theta);
  17. ts.transform.rotation.x = qtn.getX();
  18. ts.transform.rotation.y = qtn.getY();
  19. ts.transform.rotation.z = qtn.getZ();
  20. ts.transform.rotation.w = qtn.getW();
  21. pub.sendTransform(ts);
  22. }
  23. int main(int argc, char *argv[])
  24. {
  25. setlocale(LC_ALL, "");
  26. ros::init(argc, argv, "dynamic_pub");
  27. ros::NodeHandle n;
  28. ros::Subscriber sub = n.subscribe("/turtle1/pose", 100, doPose);
  29. ros::spin();
  30. return 0;
  31. }

多坐标变换

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

launch文件调用static_transform_publisher发布静态坐标变换

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

rviz

 订阅方实现

  1. #include "ros/ros.h"
  2. #include "tf2_ros/transform_listener.h"
  3. #include "tf2_ros/buffer.h"
  4. #include "tf2_geometry_msgs/tf2_geometry_msgs.h"
  5. #include "geometry_msgs/PointStamped.h"
  6. #include "geometry_msgs/TransformStamped.h"
  7. int main(int argc, char *argv[])
  8. {
  9. setlocale(LC_ALL, "");
  10. ros::init(argc, argv, "multi_tf_sub");
  11. ros::NodeHandle n;
  12. tf2_ros::Buffer buf;
  13. tf2_ros::TransformListener sub(buf);
  14. geometry_msgs::PointStamped pAtSon1; //创建son1下坐标点
  15. pAtSon1.header.stamp = ros::Time::now();
  16. pAtSon1.header.frame_id = "son1";
  17. pAtSon1.point.x = 1;
  18. pAtSon1.point.y = 1;
  19. pAtSon1.point.z = 1;
  20. ros::Rate rate(10);
  21. while(ros::ok()){
  22. try
  23. {
  24. geometry_msgs::TransformStamped son1Toson2 = buf.lookupTransform("son2", "son1", ros::Time(0)); //son2:目标坐标系 son1:原坐标系 ros::Time(0):取间隔最短的两个坐标关系帧计算相对关系
  25. ROS_INFO("原坐标系:%s, 现坐标系:%s, 偏移:%.2f, %.2f, %.2f",
  26. son1Toson2.child_frame_id.c_str(),
  27. son1Toson2.header.frame_id.c_str(),
  28. son1Toson2.transform.translation.x,
  29. son1Toson2.transform.translation.y,
  30. son1Toson2.transform.translation.z);
  31. geometry_msgs::PointStamped pAtSon2 = buf.transform(pAtSon1, "son2");
  32. ROS_INFO("所属坐标系:%s, x:%.2f, y:%.2f, z:%.2f",
  33. pAtSon2.header.frame_id.c_str(),
  34. pAtSon2.point.x,
  35. pAtSon2.point.y,
  36. pAtSon2.point.z);
  37. }
  38. catch(const std::exception& e)
  39. {
  40. ROS_INFO("\033[31m 错误提示:%s \033[0m", e.what());
  41. }
  42. rate.sleep();
  43. ros::spinOnce();
  44. }
  45. return 0;
  46. }
  1. [ INFO] [1652065692.312667393]: 错误提示:"son2" passed to lookupTransform argument target_frame does not exist.
  2. [ INFO] [1652065692.412696461]: 错误提示:"son2" passed to lookupTransform argument target_frame does not exist.
  3. [ INFO] [1652065692.512686668]: 错误提示:"son2" passed to lookupTransform argument target_frame does not exist.
  4. [ INFO] [1652065692.612818055]: 原坐标系:son1, 现坐标系:son2, 偏移:2.00, 0.00, 0.00
  5. [ INFO] [1652065692.612945493]: 所属坐标系:son2, x:3.00, y:1.00, z:1.00
  6. [ INFO] [1652065692.712710755]: 原坐标系:son1, 现坐标系:son2, 偏移:2.00, 0.00, 0.00
  7. [ INFO] [1652065692.712834071]: 所属坐标系:son2, x:3.00, y:1.00, z:1.00
  8. [ INFO] [1652065692.812684046]: 原坐标系:son1, 现坐标系:son2, 偏移:2.00, 0.00, 0.00

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号