第七课 ROS的空间描述和变换
在命令行工具中也有一个与transformcaster相类似的工具叫做static_transform_publisher,它能够接受命令行参数来接受位置信息、旋转信息、父框架、子框架以及周期信息,通常在launch文件中来指定。static_transform_publisher主要是变换两个固定坐标系之间的转换,eg:base_link和base_laser是底座与激光雷达之间的变换。
eg:
<node pkg="tf" type="static_transform_publisher" name="broadcaster" args="1 0 0 0 0 0 1 parent child 100">
下面来实现tf的订阅者
新建一个源文件turtle_tf_listener.cpp
在这个订阅者的任务里面主要是调用一个服务再生一个turtle,然后再发布cmd_vel这个主题在turtle2上面用于控制turtle2的轨迹,同时turtle2的轨迹是由transform_listener中得到的transform是由turtle1到turtle2的变换。
#include<ros/ros.h>
#include<tf/transform_listener.h>
#include<geometry_msg/Twist.h>
#include<turtlesim/Spawn.h>
int main(int argc,char **argv)
{
ros::init(argc,argv,"turtle_tf_listener");//节点名字turtle_tf_listener
ros::NodeHandle node;
//下面写服务,等待spawn服务,如果服务没有到达就一直等待
ros::service::waitForService("spawn");
//创建一个服务的客户端,其类型为turtlesim::Spawn,服务类型为spawn
ros::ServiceClient add_turtle=node.service<turtlesim::Spawn>("spawn");
//创建一个turtlesim::Spawn变量
turtlesim::Spawn srv;
//调用该服务
add_turtle.call(srv);
//下面来写一个publisher,其类型为geometry_msgs::Twist,发布到turtle2/cmd_vel上,消息队列为10
ros::Pubisher turtle_vel=node.advertise<geometry_msgs::Twist>("turtle2/cmd_vel",10);
//下面创建一个transform_listener对象
tf::transformListener listener;
//指定一个频率
ros::Rate rate(10.0);
//判断节点是否被关闭
while(ros::ok())
{
//创建一个stampedtransform对象来储存变换的信息
tf::StampedTransform transform;
try //在try里面使用lookuptransform
{
//其参数使targetframe,sourceframe,time,以及transform它用来储存变换的信息。父框架是turtle2,子框架是turtle1,时间为ros::Time(0)是指把最新的有效信息发布出去,在之前的broadcater中用的是time.now()是把现在的信息发送出去。
listener.lookupTransform("/turtle2","/turtle1",ros::Time(0),transform);
}
catch(tf::TransformException &ex)
{
ROS_ERROR("%s",ex.what());//把异常显示在终端上
ros::Duration(1.0).sleep();//显示的时间为1s。
continue;
}
//从transform中得到一个Twist的消息,然后发布给turtle2,下面实例化一个geometry_msgs
geometry_msgs::Twist vel_msg;
//角速度z的值,先指定一个系数4.0,然后其计算方法为atan2。
vel_msg.angular.z = 4.0* atan2(transform.getOrigin().y(), transform.getOrigin().x());
//线速度只指定x,系数设为0.5.其计算方法是使用平方函数。
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 1;
}
回顾一下上面程序包含了哪些内容:
首先,初始化节点,然后调用Spawn服务再生一只乌龟,接着再定义一个publisher发布消息在turtle2/cmd_vel上面,发布的消息为geometry_msg:Twist类型的vel_msg,其内容为transform储存的turtle2到turtle1的变换。
下面到CMakeList.txt里面添加内容
add_executable(turtle_tf_listener src/turtle_tf_listener.cpp)
target_link_libraries(turtle_tf_listener ${catkin_LIBRARIES})
下面再来写一个launch文件。
demo.launch
<launch>
<node pkg="turtlesim" type="turtlesim_node" name="sim"/>
<node pkg="turtlesim" type="turtle_teleop_key" name="teleop" output="screen"/>
<node pkg="learning_tf" type="turtle_tf_broadcaster" args="/turtle1" name="turtle1_tf">
<node pkg="learning_tf" type="turtle_tf_broadcaster" args="/turtle2" name="turtle2_tf">//将/turtle1和/turtle2两个的在世界坐标系中的速度和位置信息发布到tf树上
<node pkg="learning_tf" type="turtle_tf_listener" name="listener">//用来获取/turtle1与/turtle2之间的变换.
</launch>
launch文件的逻辑为首先启动turtlesim_node,然后再启动turtle_teleop_key通过按键来控制它的运动,然后再learning_tf中再生一只乌龟,有两只乌龟,在tf_listener中会发布turtle2/cmd_vel这个主题来控制turtle2的运动,在turtle_tf_broadcaster中同时启动两个节点,一个节点订阅的是turtle1的pose,另一个节点订阅的是turtle2的pose,然后在tf树中广播turtle1和turtle2的位置;然后listener订阅在世界坐标系中的turtle1个turtle2的相对位置,储存在transform中,通过transform中存储的位置信息转化为控制信息。
args是指从命令行输入的参数!!
下面去编译,别忘了增加tf的依赖项。!!!!!!!!
运行roslaunch learning_tf demo.launch
运行仿真工具rosrun rviz rviz
运行之后把tf添加进去。记得把Fixed frame选择为world。