“那就自己沉淀下来!”
本节综合TF知识,实现一只乌龟跟随另一只乌龟的任务。
I、准备工作
延用工作空间test2_ws;
功能包demo2_trans_pose存放代码;
新建乌龟的代码放在03creat_turtle.cpp中;
广播乌龟位置信息代码放在03pub_pose.cpp中。
接受广播,完成坐标转换,发布第二只乌龟的信息放在03sub_pose.cpp中。
我们的思路:乌龟中有三个坐标系,world,turtle1,turtle2。1创建turtle2用来跟随,2将turtle1和turtle2的坐标信息(实际上就是turtle1和turtle2之间的坐标系信息)传到广播器,3订阅到广播器内容后,完成坐标转换。
II、创建乌龟程序的思路
- 第一种方法是在ROS基础知识中学到的,通过rosservice调用/spawn来生成
rosservice call /spawn "x: 1.0
y: 1.0
theta: 1.0
name: 'turtle_flow'"
-第二种是用程序生成,然后编写客户端调用服务
ros::service::waitForService("/spawn");
turtlesim::Spawn spawn;
spawn.request.name = "turtle2";
spawn.request.x = 1.0;
spawn.request.y = 2.0;
spawn.request.theta = 3.12415926;
bool flag = client.call(spawn);
III、广播乌龟位姿程序的思路
- 首先订阅两只乌龟的位置信息话题,得到它们坐标:
ros::Subscriber sub = nh.subscribe<turtlesim::Pose>(turtle_name + "/pose",1000,doPose);
- 分别进入回调函数完成常规的广播操作
这里注意此时发送时坐标系都是world坐标系,对应的子坐标系才是自己名字。
static tf2_ros::TransformBroadcaster broadcaster;
// 6-2.将 pose 信息转换成 TransFormStamped
geometry_msgs::TransformStamped tfs;
tfs.header.frame_id = "world";
3.填充数据然后发送
broadcaster.sendTransform(tfs);
IV、接受乌龟广播,坐标变换
- 创建TF订阅对象
tf2_ros::Buffer buffer;
tf2_ros::TransformListener listener(buffer);
- 创建控制第二只乌龟的发布器
ros::Publisher pub = nh.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel",1000);
- 获取两个坐标系的相对信息(注意顺序)
这里的ros::Time(0),表示从最近的信息开始提取,而不采用now,如果用now会有很大延迟。这里是将乌龟2的坐标映射到turtle1坐标系上。这样turtle1永远是(0,0)。(子坐标系,父坐标系)
geometry_msgs::TransformStamped tfs = buffer.lookupTransform("turtle2","turtle1",ros::Time(0));
- 填充乌龟2信息实现发布:
geometry_msgs::Twist twist;
twist.linear.x = 0.5*sqrt(pow(tfs.transform.translation.x,2) + pow(tfs.transform.translation.y,2));
twist.angular.z = 5 * atan2(tfs.transform.translation.y, tfs.transform.translation.x);
pub.publish(twist);
turtle2在turtle1坐标系下,方向取正切加系数。
V、launch文件编写
实现流程:
- 运行turtlesim_node
- 运行turtle_deleop
- 运行创建乌龟2节点
- 运行我们的发布节点,发布乌龟1的信息,再发布乌龟2的信息
- 运行我们的收听节点,控制乌龟2
简单起见,我们封装在一个launch文件中:
<launch>
<!-- 启动乌龟节点与键盘控制节点 -->
<node pkg="turtlesim" type="turtlesim_node" name="turtle1" output="screen" />
<node pkg="turtlesim" type="turtle_teleop_key" name="key_control" launch-prefix="xterm -e"/>
<!-- 启动创建第二只乌龟的节点 -->
<node pkg="demo2_trans_pose" type="creat_turtle2" name="turtle2" output="screen" />
<!-- 启动两个坐标发布节点 -->
<node pkg="demo2_trans_pose" type="pub_tf_turtle" name="pub_turtle1" args="turtle1" output="screen" />
<node pkg="demo2_trans_pose" type="pub_tf_turtle" name="pub_turtle2" args="turtle2" output="screen" />
<!-- 启动坐标转换节点 -->
<node pkg="demo2_trans_pose" type="sub_tf_turtle" name="sub_follow" output="screen" />
</launch>
然后打开rviz ,添加tf坐标,全局坐标选择world,控制乌龟运动,结果如下:
打开rviz,如下
VI 全部源码
03creat_turtle.cpp
/*
创建第二只小乌龟
*/
#include "ros/ros.h"
#include "turtlesim/Spawn.h"
int main(int argc, char *argv[])
{
setlocale(LC_ALL,"");
//执行初始化
ros::init(argc,argv,"create_turtle2");
//创建节点
ros::NodeHandle nh;
//创建服务客户端
ros::ServiceClient client = nh.serviceClient<turtlesim::Spawn>("/spawn");
ros::service::waitForService("/spawn");
turtlesim::Spawn spawn;
spawn.request.name = "turtle2";
spawn.request.x = 1.0;
spawn.request.y = 2.0;
spawn.request.theta = 3.12415926;
bool flag = client.call(spawn);
if (flag)
{
ROS_INFO("乌龟%s创建成功!",spawn.response.name.c_str());
}
else
{
ROS_INFO("乌龟2创建失败!");
}
ros::spin();
return 0;
}
03pub_pose.cpp
/*
该文件实现:需要订阅 turtle1 和 turtle2 的 pose,然后广播相对 world 的坐标系信息
注意: 订阅的两只 turtle,除了命名空间(turtle1 和 turtle2)不同外,
其他的话题名称和实现逻辑都是一样的,
所以我们可以将所需的命名空间通过 args 动态传入
实现流程:
1.包含头文件
2.初始化 ros 节点
3.解析传入的命名空间
4.创建 ros 句柄
5.创建订阅对象
6.回调函数处理订阅的 pose 信息
6-1.创建 TF 广播器
6-2.将 pose 信息转换成 TransFormStamped
6-3.发布
7.spin
*/
//1.包含头文件
#include "ros/ros.h"
#include "turtlesim/Pose.h"
#include "tf2_ros/transform_broadcaster.h"
#include "tf2/LinearMath/Quaternion.h"
#include "geometry_msgs/TransformStamped.h"
//保存乌龟名称
std::string turtle_name;
void doPose(const turtlesim::Pose::ConstPtr& pose)
{
// 6-1.创建 TF 广播器 ---------------------------------------- 注意 static
static tf2_ros::TransformBroadcaster broadcaster;
// 6-2.将 pose 信息转换成 TransFormStamped
geometry_msgs::TransformStamped tfs;
tfs.header.frame_id = "world";
tfs.header.stamp = ros::Time::now();
tfs.child_frame_id = turtle_name;
tfs.transform.translation.x = pose->x;
tfs.transform.translation.y = pose->y;
tfs.transform.translation.z = 0.0;
tf2::Quaternion qtn;
qtn.setRPY(0,0,pose->theta);
tfs.transform.rotation.x = qtn.getX();
tfs.transform.rotation.y = qtn.getY();
tfs.transform.rotation.z = qtn.getZ();
tfs.transform.rotation.w = qtn.getW();
// 6-3.发布
broadcaster.sendTransform(tfs);
}
int main(int argc, char *argv[])
{
setlocale(LC_ALL,"");
// 2.初始化 ros 节点
ros::init(argc,argv,"pub_turtle_tf");
// 3.解析传入的命名空间
if (argc != 2)
{
ROS_ERROR("请传入正确的参数");
}
else
{
turtle_name = argv[1];
ROS_INFO("乌龟 %s 坐标发送启动",turtle_name.c_str());
}
// 4.创建 ros 句柄
ros::NodeHandle nh;
// 5.创建订阅对象
ros::Subscriber sub = nh.subscribe<turtlesim::Pose>(turtle_name + "/pose",1000,doPose);
// 6.回调函数处理订阅的 pose 信息
// 6-1.创建 TF 广播器
// 6-2.将 pose 信息转换成 TransFormStamped
// 6-3.发布
// 7.spin
ros::spin();
return 0;
}
03sub_pose.cpp
/*
订阅 turtle1 和 turtle2 的 TF 广播信息,查找并转换时间最近的 TF 信息
将 turtle1 转换成相对 turtle2 的坐标,在计算线速度和角速度并发布
实现流程:
1.包含头文件
2.初始化 ros 节点
3.创建 ros 句柄
4.创建 TF 订阅对象
5.处理订阅到的 TF
6.spin
*/
//1.包含头文件
#include "ros/ros.h"
#include "tf2_ros/transform_listener.h"
#include "geometry_msgs/TransformStamped.h"
#include "geometry_msgs/Twist.h"
int main(int argc, char *argv[])
{
setlocale(LC_ALL,"");
// 2.初始化 ros 节点
ros::init(argc,argv,"sub_turtle_tf");
// 3.创建 ros 句柄
ros::NodeHandle nh;
// 4.创建 TF 订阅对象
tf2_ros::Buffer buffer;
tf2_ros::TransformListener listener(buffer);
// 5.处理订阅到的 TF
// 需要创建发布 /turtle2/cmd_vel 的 publisher 对象
ros::Publisher pub = nh.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel",1000);
ros::Rate rate(10);
while (ros::ok())
{
try
{
//5-1.先获取 turtle1 相对 turtle2 的坐标信息
geometry_msgs::TransformStamped tfs = buffer.lookupTransform("turtle2","turtle1",ros::Time(0));
//5-2.根据坐标信息生成速度信息 -- geometry_msgs/Twist.h
geometry_msgs::Twist twist;
twist.linear.x = 0.5*sqrt(pow(tfs.transform.translation.x,2) + pow(tfs.transform.translation.y,2));
twist.angular.z = 5 * atan2(tfs.transform.translation.y, tfs.transform.translation.x);
//5-3.发布速度信息 -- 需要提前创建 publish 对象
pub.publish(twist);
}
catch(const std::exception& e)
{
// std::cerr << e.what() << '\n';
ROS_INFO("错误提示:%s",e.what());
}
rate.sleep();
// 6.spin
ros::spinOnce();
}
return 0;
}