ROS2中的TF(Transform)系统:机器人坐标系的管理神器

ROS 2 TF 概述

TF(Transform) 是ROS中用于跟踪多个坐标系之间变换关系的库。在ROS 2中,TF系统被重构为 TF2,提供了更高效、更灵活的坐标变换管理。


核心概念

1. 坐标系(Frame)

  • 每个机器人部件、传感器或环境物体都有自己的坐标系
  • 例如:base_link(机器人基座)、laser(激光雷达)、camera(相机)

2. 变换(Transform)

  • 描述两个坐标系之间的平移(translation)和旋转(rotation)关系
  • 表示为:frame_B 相对于 frame_A 的位置和姿态

3. 变换树(Transform Tree)

  • 所有坐标系通过父子关系连接成一棵树
  • 必须有一个根坐标系(通常是mapodom

ROS 2 TF2 架构

┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐ │ TF2 广播者 │ │ TF2 缓冲区 │ │ TF2 监听者 │ │ (Broadcaster) │────▶│ (Buffer) │◀────│ (Listener) │ └─────────────────┘ └─────────────────┘ └─────────────────┘ │ │ ▼ ▼ ┌─────────────────┐ ┌─────────────────┐ │ 发布变换信息 │ │ 查询坐标变换 │ │ /tf 话题 │ │ lookupTransform │ └─────────────────┘ └─────────────────┘ 

核心组件

组件功能对应类/节点
Broadcaster广播坐标变换TransformBroadcaster
Static Broadcaster广播静态变换(固定关系)StaticTransformBroadcaster
Listener监听并查询变换Buffer + TransformListener
Buffer存储变换历史tf2_ros::Buffer

代码示例

1. 发布动态变换(Dynamic Transform)

import rclpy from rclpy.node import Node from geometry_msgs.msg import TransformStamped import tf2_ros import math classDynamicFramePublisher(Node):def__init__(self):super().__init__('dynamic_frame_publisher')# 创建TF广播者 self.br = tf2_ros.TransformBroadcaster(self)# 定时发布 self.timer = self.create_timer(0.1, self.publish_transform) self.t =0.0defpublish_transform(self): t = TransformStamped() t.header.stamp = self.get_clock().now().to_msg() t.header.frame_id ='base_link'# 父坐标系 t.child_frame_id ='rotating_laser'# 子坐标系# 旋转运动 self.t +=0.1 t.transform.translation.x =1.0 t.transform.translation.y =0.0 t.transform.translation.z =0.5# 四元数表示旋转 t.transform.rotation.x =0.0 t.transform.rotation.y =0.0 t.transform.rotation.z = math.sin(self.t /2) t.transform.rotation.w = math.cos(self.t /2) self.br.sendTransform(t)defmain(): rclpy.init() node = DynamicFramePublisher() rclpy.spin(node) rclpy.shutdown()

2. 发布静态变换(Static Transform)

from tf2_ros.static_transform_broadcaster import StaticTransformBroadcaster classStaticFramePublisher(Node):def__init__(self):super().__init__('static_frame_publisher') self.static_br = StaticTransformBroadcaster(self)# 只需要发布一次 static_transform = TransformStamped() static_transform.header.stamp = self.get_clock().now().to_msg() static_transform.header.frame_id ='base_link' static_transform.child_frame_id ='camera_link' static_transform.transform.translation.x =0.5 static_transform.transform.translation.y =0.0 static_transform.translation.z =0.3# 相机朝向(旋转90度) static_transform.transform.rotation.x =0.0 static_transform.transform.rotation.y =0.0 static_transform.transform.rotation.z =0.707 static_transform.transform.rotation.w =0.707 self.static_br.sendTransform(static_transform)

3. 监听变换(Transform Listener)

import tf2_ros from tf2_ros import LookupException, ConnectivityException, ExtrapolationException classFrameListener(Node):def__init__(self):super().__init__('frame_listener')# 创建Buffer和Listener self.tf_buffer = tf2_ros.Buffer() self.tf_listener = tf2_ros.TransformListener(self.tf_buffer, self) self.timer = self.create_timer(1.0, self.lookup_transform)deflookup_transform(self):try:# 查询从 'base_link' 到 'laser' 的变换 transform = self.tf_buffer.lookup_transform('base_link',# 目标坐标系'laser',# 源坐标系 rclpy.time.Time()# 最新时间) self.get_logger().info(f"Translation: [{transform.transform.translation.x:.2f}, "f"{transform.transform.translation.y:.2f}, "f"{transform.transform.translation.z:.2f}]")except(LookupException, ConnectivityException, ExtrapolationException)as e: self.get_logger().warn(f'Could not transform: {str(e)}')

常用工具

命令行工具

# 查看当前TF树 ros2 run tf2_tools view_frames # 查询特定变换 ros2 run tf2_ros tf2_echo base_link laser # 发布静态变换 ros2 run tf2_ros static_transform_publisher 001000 base_link camera 

RViz2 可视化

在RViz2中添加 TF 显示插件,可以实时可视化坐标系:

  • 红色 = X轴
  • 绿色 = Y轴
  • 蓝色 = Z轴

ROS 1 vs ROS 2 TF 对比

特性ROS 1 TFROS 2 TF2
默认库tftf2
时间处理较简单支持时间旅行(查询历史变换)
数据类型专用消息与ROS 2消息系统更好集成
静态变换混合处理独立话题 /tf_static
性能一般更高效,支持零拷贝
Python支持较弱原生支持,API更友好

最佳实践

  1. 区分静态/动态变换:固定关系用StaticTransformBroadcaster,减少网络负载
  2. 保持时间同步:确保所有节点使用相同的时间源(ROS时间 vs 系统时间)
  3. 异常处理:查询变换时总是捕获可能的异常
  4. 命名规范:使用REP 105标准坐标系命名(mapodombase_link
  5. 避免环形依赖:TF树必须是严格的树形结构,不能闭环

典型应用场景

  • 传感器数据融合:将激光雷达、相机数据转换到统一坐标系
  • 导航规划:将目标点从地图坐标系转换到机器人坐标系
  • 机械臂控制:计算末端执行器相对于基座的位姿
  • 多机器人协作:统一不同机器人的坐标参考系

Read more

前端防范 XSS(跨站脚本攻击)

目录 一、防范措施 1.layui util  核心转义的特殊字符 示例 2.js-xss.js库 安装 1. Node.js 环境(npm/yarn) 2. 浏览器环境 核心 API 基础使用 1. 基础过滤(默认规则) 2. 自定义过滤规则 (1)允许特定标签 (2)允许特定属性 (3)自定义标签处理 (4)自定义属性处理 (5)转义特定字符 常见场景示例 1. 过滤用户输入的评论内容 2. 允许特定富文本标签(如富文本编辑器内容) 注意事项 更多配置 XSS(跨站脚本攻击)是一种常见的网络攻击手段,它允许攻击者将恶意脚本注入到其他用户的浏览器中。

详细教程:如何从前端查看调用接口、传参及返回结果(附带图片案例)

详细教程:如何从前端查看调用接口、传参及返回结果(附带图片案例)

目录 1. 打开浏览器开发者工具 2. 使用 Network 面板 3. 查看具体的API请求 a. Headers b. Payload c. Response d. Preview e. Timing 4. 实际操作步骤 5. 常见问题及解决方法 a. 无法看到API请求 b. 请求失败 c. 跨域问题(CORS) 作为一名后端工程师,理解前端如何调用接口、传递参数以及接收返回值是非常重要的。下面将详细介绍如何通过浏览器开发者工具(F12)查看和分析这些信息,并附带图片案例帮助你更好地理解。 1. 打开浏览器开发者工具 按下 F12 或右键点击页面选择“检查”可以打开浏览器的开发者工具。常用的浏览器如Chrome、Firefox等都内置了开发者工具。下面是我选择我的一篇文章,打开开发者工具进行演示。 2. 使用

Cursor+Codex隐藏技巧:用截图秒修前端Bug的保姆级教程(React/Chakra UI案例)

Cursor+Codex隐藏技巧:用截图秒修前端Bug的保姆级教程(React/Chakra UI案例) 前端开发中最令人头疼的莫过于那些难以定位的UI问题——元素错位、样式冲突、响应式失效...传统调试方式往往需要反复修改代码、刷新页面、检查元素。现在,通过Cursor编辑器集成的Codex功能,你可以直接用截图交互快速定位和修复这些问题。本文将带你从零开始,掌握这套革命性的调试工作流。 1. 环境准备与基础配置 在开始之前,确保你已经具备以下环境: * Cursor编辑器最新版(v2.5+) * Node.js 18.x及以上版本 * React 18项目(本文以Chakra UI 2.x为例) 首先在Cursor中安装Codex插件: 1. 点击左侧扩展图标 2. 搜索"Codex"并安装 3. 登录你的OpenAI账户(需要ChatGPT Plus订阅) 关键配置项: // 在项目根目录创建.