如上图:对于同一物体在两个坐标系之间的变换是坐标变换的基本变换,可以通过旋转矩阵、四元数等多种变换表达方法进行分析,而对于ROS来说,我们就可以免去那么多的自行计算,通过TF功能包,便捷的实现这些要求。
例如,我们可以通过TF功能包快速的获得以下的内容:
- 3s前,机器人头部坐标系相对于全局坐标系的关系;
- 当前机器人末端坐标系相对于躯干中心坐标系的关系;
- 机器人躯干中心坐标系相对于全局坐标系的关系;
实现:不同于之前的通讯方法,TF功能包通过广播TF变换、监听TF变换,通过一个TF树形结构保存数据,完成最终的计算;
这里我们以一个简单的机器人举例,它由一个底座和一个激光雷达构成,如下图所示:
通过激光雷达,我们可以轻松的获得laser坐标系的各轴坐标,然后就可以通过二者之间的坐标系转换从而得到link坐标系的坐标,图示如下:
sudo apt-get install ros-noetic-turtle-tf
roslaunch turtle_tf turtle_tf_dmeo.launch
No module named yaml
问题原因是python的版本问题,可以通过命令sudo apt install python-is-python3
解决,正常打开后,界面初始会出现两只海龟,其中左下角的那只随时间跟随中间的那只,如下图所示:
rosrun turtlesim turtle_teleop_key
根据上图中左图所示,通过键盘控制中间的海龟移动后,左下角的海龟会自动不断调整自身运动方向,最后达到与turtle1相同的位置,其中就涉及到了TF坐标变换的作用。
rosrun tf view_frames
PDF文件:存在三个坐标系:世界坐标系为海龟界面的左下角(固定坐标系)、turtle1、turtle2三个坐标系;中间只要有箭头,就说明联系成功,可以通过查询树形机构实现得到各坐标关系;
2)TF树查询关系:直接实时查询树中任意两个坐标系之间的关系:translation平移、rotation旋转
rosrun tf tf_echo turtle1 turtle2
rosrun rviz rviz -d `rospack find turtle_tf`/rviz/turtle_rviz.rviz
catkin_create_pkg learning_tf roscpp rospy tf turtlesim
/**
* @file turtle_tf_broadcaster.cpp
* @author zhaoxu
* @brief 该例程产生tf数据,并计算发布turtle2的速度指令
* @version 0.1
* @date 2022-04-24
*
* @copyright Copyright (c) 2022
*
*/
#include
#include
#include
std::string turtle_name;
void poseCallback(const turtlesim::PoseConstPtr& msg)
{
// 创建tf的广播器
static tf::TransformBroadcaster br;
// 初始化tf数据
tf::Transform transform;
// 平移参数xyz
transform.setOrigin(tf::Vector3(msg->x, msg->y, 0.0));
// 旋转,四元数
tf::Quaternion q;
q.setRPY(0, 0, msg->theta);
transform.setRotation(q);
// 广播world与海龟坐标系之间的tf数据
br.sendTransform(tf::StampedTransform(transform, ros::Time::now(), "world", turtle_name));
}
int main(int argc, char** argv)
{
// 初始化ROS节点
ros::init(argc, argv, "my_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;
}
/**
* @file turtle_tf_listener.cpp
* @author zhaoxu
* @brief 该例程负责监听tf数据,并计算、发布turtle2的速度指令
* @version 0.1
* @date 2022-04-24
*
* @copyright Copyright (c) 2022
*
*/
#include
#include
#include
#include
int main(int argc, char** argv)
{
// 初始化ROS节点
ros::init(argc, argv, "my_tf_listener");
// 创建节点句柄
ros::NodeHandle node;
// 请求产生turtle2
ros::service::waitForService("/spawn");
ros::ServiceClient add_turtle = node.serviceClient<turtlesim::Spawn>("/spawn");
turtlesim::Spawn srv;
add_turtle.call(srv);
// 创建发布turtle2速度控制指令的发布者
ros::Publisher turtle_vel = node.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel", 10);
// 创建tf的监听器
tf::TransformListener listener;
ros::Rate rate(10.0);
while (node.ok())
{
// 获取turtle1与turtle2坐标系之间的tf数据
tf::StampedTransform transform;
try
{
// 等待变换
listener.waitForTransform("/turtle2", "/turtle1", ros::Time(0), ros::Duration(3.0));
// 查询坐标系关系
listener.lookupTransform("/turtle2", "/turtle1", ros::Time(0), transform);
}
catch(tf::TransformException &ex)
{
ROS_ERROR("%s", ex.what());
ros::Duration(1.0).sleep();
continue;
}
// 根据turtle1与turtle2坐标系之间的位置关系,发布turtle2的速度控制指令
geometry_msgs::Twist vel_msg;
vel_msg.angular.z = 4.0 * atan2(transform.getOrigin().y(),
transform.getOrigin().x());
vel_msg.linear.x = 0.5 * sqrt(pow(transform.getOrigin().x(), 2) +
pow(transform.getOrigin().y(), 2));
turtle_vel.publish(vel_msg);
rate.sleep();
}
return 0;
}
# 设置需要编译的代码和生成的可执行文件
add_executable(turtle_tf_broadcaster src/turtle_tf_broadcaster.cpp)
# 设置链接库
target_link_libraries(turtle_tf_broadcaster ${catkin_LIBRARIES})
# 设置需要编译的代码和生成的可执行文件
add_executable(turtle_tf_listener src/turtle_tf_listener.cpp)
# 设置链接库
target_link_libraries(turtle_tf_listener ${catkin_LIBRARIES})
cd ~/catkin_ws
catkin_make
roscore
rosrun turtlesim turtlesim_node
rosrun learning_tf turtle_tf_broadcaster __name:=turtle1_tf_broadcaster /turtle1
rosrun learning_tf turtle_tf_broadcaster __name:=turtle2_tf_broadcaster /turtle2
rosrun learning_tf turtle_tf_listener
rosrun turtlesim turtle_teleop_key
最终的效果与之前的demo类似,这里不予图示。
本节博客主要介绍了TF坐标管理系统,从机器人的基本坐标变换出发,介绍了TF功能包的作用,并介绍了多种可视化方法,最后通过C++编程,自行实现了对于海龟跟踪demo的模拟,ROS入门教程还剩最后一篇,会对于ROS的功能进行总结,并介绍后续的学习方法,敬请期待。