ROS-TF库-写一个TransformBroadcaster发布

本文详细介绍了在ROS中如何使用TransformBroadcaster类发布坐标变换。通过实例代码演示了从订阅/turtle_pose话题获取数据,到创建并发布tf变换的全过程。文章适合ROS初学者及对tf变换感兴趣的开发者。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1 TransformBroadcaster类

  • tf包提供了TransformBroadcaster类的实现,以帮助简化tf发布转换的任务。
  • 要使用TransformBroadcaster,我们需要包含<tf/transform_Broadcaster.h>头文件。
    如下是发布一个world-turtle1的一个坐标变换,来具体看TransformBroadcaster类的用法:
#include <ros/ros.h>
#include <tf/transform_broadcaster.h>
#include <turtlesim/Pose.h>
std::string turtle_name;

void poseCallback(const turtlesim::PoseConstPtr& msg){

  // 1. 创建一个tf发布对象
  static tf::TransformBroadcaster br;

  // 2. 创建一个tf对象
  tf::Transform transform;
  
  // 3. 设置tf的平移旋转,(把小龟的2d位姿复制到3d变换)
  transform.setOrigin( tf::Vector3(msg->x, msg->y, 0.0) );
  tf::Quaternion q;
  q.setRPY(0, 0, msg->theta);
  transform.setRotation(q);

  // 4. 发布tf变换
  br.sendTransform(tf::StampedTransform(transform, ros::Time::now(), "world", turtle_name));
}

int main(int argc, char** argv){
  ros::init(argc, argv, "turtle_tf_broadcaster");
  if (argc != 2)
  {ROS_ERROR("need turtle name as argument"); return -1; };
  turtle_name = argv[1];
  ros::NodeHandle node;
  ros::Subscriber sub = node.subscribe(turtle_name+"/pose", 10, &poseCallback);
  ros::spin();
  return 0;
};


2 运行发布

  • 程序依赖/turtle_pose话题,启动一个turtlesim后再运行此程序,结果如下.
    发布一个tf

  • 显示tf树
     rosrun rqt_tf_tree rqt_tf_tree

  • 输出两个坐标系的tf
     rosrun tf tf_echo /co1 /co2

### 如何在 ROS发布 TF 变换 #### 使用 `static_transform_publisher` 发布静态变换ROS 中,可以通过命令行工具 `static_transform_publisher` 来发布静态的坐标变换关系。该工具属于 `tf2_ros` 功能包的一部分[^2]。下面是一个典型的例子: ```bash rosrun tf2_ros static_transform_publisher 5 5 5 0 0 0 /base_link /radar ``` 上述命令表示创建了一个从 `/base_link` 到 `/radar` 的固定变换关系,其中 `(5, 5, 5)` 是平移量 (单位为米),`(0, 0, 0)` 表示旋转角度 (以弧度制给出)[^1]。 --- #### 编 Python 或 C++ 节点来发布动态变换 对于更复杂的场景,可以编自定义节点来实时计算并广播动态变换数据。以下是基于 Python 和 C++ 的实现方式: ##### 基于 Python 的动态变换发布器 以下代码展示了如何通过 Python 广播一个随时间变化的坐标系转换关系: ```python #!/usr/bin/env python import rospy import tf2_ros from geometry_msgs.msg import TransformStamped import math def broadcast_tf(): node = rospy.init_node('dynamic_tf_broadcaster', anonymous=True) br = tf2_ros.TransformBroadcaster() rate = rospy.Rate(10) while not rospy.is_shutdown(): t = rospy.Time.now().to_sec() * math.pi transform = TransformStamped() transform.header.stamp = rospy.Time.now() transform.header.frame_id = "world" transform.child_frame_id = "robot" # 设置平移部分 transform.transform.translation.x = math.sin(t) transform.transform.translation.y = math.cos(t) transform.transform.translation.z = 0.0 # 设置四元数形式的旋转部分 transform.transform.rotation.x = 0.0 transform.transform.rotation.y = 0.0 transform.transform.rotation.z = 0.0 transform.transform.rotation.w = 1.0 br.sendTransform(transform) rate.sleep() if __name__ == '__main__': try: broadcast_tf() except rospy.ROSInterruptException: pass ``` 这段脚本会每隔一段时间更新一次 `/world` 到 `/robot` 的变换矩阵,并将其发送到 TF 主题上[^3]。 --- ##### 基于 C++ 的动态变换发布器 如果偏好使用 C++ 实现,则可参考如下代码片段: ```cpp #include <ros/ros.h> #include <tf2/LinearMath/Quaternion.h> #include <geometry_msgs/TransformStamped.h> #include <tf2_ros/transform_broadcaster.h> int main(int argc, char** argv){ ros::init(argc, argv, "my_tf_broadcaster"); ros::NodeHandle nh; tf2_ros::TransformBroadcaster broadcaster; ros::Rate rate(10); while(nh.ok()){ geometry_msgs::TransformStamped tfs; tfs.header.stamp = ros::Time::now(); tfs.header.frame_id = "map"; tfs.child_frame_id = "drone"; // 定义位置偏移 tfs.transform.translation.x = cos(ros::Time::now().toSec()); tfs.transform.translation.y = sin(ros::Time::now().toSec()); tfs.transform.translation.z = 0.0; // 定义姿态角 tf2::Quaternion q; q.setRPY(0, 0, ros::Time::now().toSec()); // 绕 Z 轴旋转一定角度 tfs.transform.rotation.x = q.getX(); tfs.transform.rotation.y = q.getY(); tfs.transform.rotation.z = q.getZ(); tfs.transform.rotation.w = q.getW(); broadcaster.sendTransform(tfs); rate.sleep(); } } ``` 以上程序实现了类似的逻辑——即让无人机围绕某个中心点做圆周运动的同时调整其朝向[^4]。 --- #### 总结 无论是简单的静态变换还是复杂的时间依赖型动态变换,在 ROS 系统里都可以借助内置的功能包轻松完成设置工作。具体方法取决于实际需求以及开发者的熟悉程度。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值