【ROS程序】 --- 2-3. TF--实现乌龟跟随

本节综合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、广播乌龟位姿程序的思路

  1. 首先订阅两只乌龟的位置信息话题,得到它们坐标:
ros::Subscriber sub = nh.subscribe<turtlesim::Pose>(turtle_name + "/pose",1000,doPose);
  1. 分别进入回调函数完成常规的广播操作
    这里注意此时发送时坐标系都是world坐标系,对应的子坐标系才是自己名字。
	static tf2_ros::TransformBroadcaster broadcaster;
    //  6-2.将 pose 信息转换成 TransFormStamped
    geometry_msgs::TransformStamped tfs;
    tfs.header.frame_id = "world";

3.填充数据然后发送

 broadcaster.sendTransform(tfs);

IV、接受乌龟广播,坐标变换

  1. 创建TF订阅对象
	tf2_ros::Buffer buffer;
    tf2_ros::TransformListener listener(buffer);
  1. 创建控制第二只乌龟的发布器
ros::Publisher pub = nh.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel",1000);
  1. 获取两个坐标系的相对信息(注意顺序)
    这里的ros::Time(0),表示从最近的信息开始提取,而不采用now,如果用now会有很大延迟。这里是将乌龟2的坐标映射到turtle1坐标系上。这样turtle1永远是(0,0)。(子坐标系,父坐标系)
geometry_msgs::TransformStamped tfs = buffer.lookupTransform("turtle2","turtle1",ros::Time(0));
  1. 填充乌龟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文件编写

实现流程:

  1. 运行turtlesim_node
  2. 运行turtle_deleop
  3. 运行创建乌龟2节点
  4. 运行我们的发布节点,发布乌龟1的信息,再发布乌龟2的信息
  5. 运行我们的收听节点,控制乌龟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;
}
  • 0
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值