醋醋百科网

Good Luck To You!

《ROS入门》第12讲ROS中的坐标系管理系统

TF功能包能干什么?

  • o 五秒钟之前,机器人头部坐标系相对于全局坐
  • 标系的关系是什么样的?
  • o 机器人夹取的物体相对于机器人中心坐标系的
  • 位置在哪里?
  • o 机器人中心坐标系相对于全局坐标系的位置在
  • 哪里?
  • TF坐标变换如何实现?

  • o 广播TF变换
  • o 监听TF变换
  • 备注:具体坐标变换的理论公式推导参考《机器人学导论》

    初体验:

    在终端启动海龟的TF坐标变换的例程,步骤:

    第一,安装TF功能包:

    sudo apt-get install ros-版本号-turtle-tf

    (我使用的是noetic,自带TF功能包)

    第二步,使用roslaunch启动

    roslaunch turtle_tf2 turtle_tf2_demo_cpp.launch 

    第三步,启动海龟键盘节点

    rosrun turtlesim turtle_teleop_key 

    然后使用键盘的上下左右键操作即可看到现象

    tf坐标系广播与监听的编程实现

    1,创建功能包

    $ cd ~/catkin_ws/src
    $ catkin_create_pkg doubi roscpp rospy tf turtlesim
    $ cd ~/catkin_ws/src
    $ catkin_make

    2,编写程序

    如何实现一个tf广播器

    • o定义TF广播器(TransformBroadcaster)
    • o创建坐标变换值;
    • o发布坐标变换(sendTransform)
    /**
     * 该例程产生tf数据,并计算、发布turtle2的速度指令
     */
    
    #include <ros/ros.h>
    #include <tf/transform_broadcaster.h>
    #include <turtlesim/Pose.h>
    
    std::string turtle_name;
    
    void poseCallback(const turtlesim::PoseConstPtr& msg)
    {
    	// 创建tf的广播器
    	static tf::TransformBroadcaster br;
    
    	// 初始化tf数据
    	tf::Transform transform;
    	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;
    };
    

    如何实现一个TF监听器

    • o定义TF监听器;
    • o查找坐标变换;
    /**
     * 该例程监听tf数据,并计算、发布turtle2的速度指令
     */
    
    #include <ros/ros.h>
    #include <tf/transform_listener.h>
    #include <geometry_msgs/Twist.h>
    #include <turtlesim/Spawn.h>
    
    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 (tf2::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;
    };
    

    3,编译并测试

    $ cd ~/catkin_ws
    $ catkin_make
    $ source devel/setup.bash
    $ roscore
    $ rosrun turtlesim turtlesim_node
    $ rosrun doubi turtle_tf_broadcaster __name:=turtle1_tf_broadcaster /turtle1
    $ rosrun doubi turtle_tf_broadcaster __name:=turtle2_tf_broadcaster /turtle2
    $ rosrun doubi turtle_tf_listener
    $ rosrun turtlesim turtle_teleop_key
    控制面板
    您好,欢迎到访网站!
      查看权限
    网站分类
    最新留言