百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

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

zhezhongyun 2025-08-03 09:25 3 浏览

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

    相关推荐

    3 分钟!AI 从零开发五子棋全过程曝光,网友:这效率我服了

    <!DOCTYPEhtml><htmllang="zh-CN"><head><metacharset="UTF-8...

    一行代码实现display&quot;过渡动画&quot;原理

    作者:Peter谭老师转发链接:https://mp.weixin.qq.com/s/XhwPOv62gypzq5MhhP-5vg写本文的起因上篇文章,提到如何让display出现过渡动画,却没有仔...

    脑洞:琼恩·雪诺、蝙蝠侠和魔形女的灵魂宠物了解一下

    AlekseiVinogradovisaRussianfreelancedigitalartistwhoshareshisskillsandtalentwith120k...

    浏览器的渲染机制、重绘、重排

    1、什么是重排和重绘网页生成过程:HTML被HTML解析器解析成DOM树css则被css解析器解析成CSSOM树结合DOM树和CSSOM树,生成一棵渲染树(RenderTree)生成布局(flo...

    托福写作高频考题写作思路&amp;词汇丨考虫独家

    科技话题与媒体话题是托福写作的常考话题很多考生对这两类话题里的专有词汇表达也许很不了解所以今天就跟随考虫托福写作老师刘云龙老师一起来学习在这些话题的写作里你可以使用哪些有用的表达。希望大家有收获!记得...

    在优麒麟上使用 Electron 开发桌面应用

    使用Web标准来创建桌面GUI,上手快、成本低、跨平台、自适应分辨率,这些都是Electron的优势。作者/来源:优麒麟Electron是由Github开发,用HTML、CSS和...

    php手把手教你做网站(三十八)jquery 转轮盘抽奖,开盲盒

    抽奖和开盲盒性质一样的都是通过ajax读取后台的随机数据。1、转轮盘本来是想直接绘图实现轮盘,但是没有找到怎么填充文字,只好把轮盘弄成了背景图,通常用于游戏抽道具,商城积分抽奖,公司年末员工抽奖点击抽...

    用 CSS 整活!3D 轮播图手把手教学,快乐代码敲出来

    兄弟们,今天咱来搞点好玩的——用CSS整一个3D轮播图!咱野生程序员就是要在代码里找乐子,技术和快乐咱都得要!代码是写不完的,但咱能自己敲出快乐来,走起!一、先整个容器,搭个舞台咋先写一个...

    实现一个超酷的 3D 立体卡片效 #前端开发

    今天我们来实现一个超酷的3D立体卡片效果。正常情况下就是一个普通的图片展示卡片,鼠标悬停的时候图片会跳出卡片,并将影子投射到背景卡片上,在视觉上有一个3D立体感。html主要分成3个部分:容器→背景层...

    Vue 3 Teleport与Suspense:解决UI难题的两个&quot;隐藏大招&quot;

    模态框的"层级噩梦"与Teleport的救赎"这个模态框怎么又被父容器截断了?"团队协作开发后台系统时,小张第N次遇到这个问题。多层嵌套的组件结构里,弹窗被overfl...

    让交互更加生动!有意思的鼠标跟随 3D 旋转动效

    今天,群友问了这样一个问题,如下所示的鼠标跟随交互效果,如何实现:简单分析一下,这个交互效果主要有两个核心:借助了CSS3D的能力元素的旋转需要和鼠标的移动相结合本文,就将讲述如何使用纯CSS...

    填坑:transform元素导致zindex失效终极方法

    今天遇到了使用css3动画的元素层级被放大置顶的问题,ios浏览器上没问题,安卓原生浏览器和安卓微信上有问题。使用了css3动画的元素z-index失效,兄弟元素设置多高的z-index都盖不住解决办...

    诡异的层级错乱:一个被transform隐藏的CSS陷阱

    周五下午三点十七分,设计部突然发来紧急截图——原本应该悬浮在顶部的导航菜单,此刻正诡异地被下方的轮播图遮挡。我盯着屏幕上错乱的层级关系,手指下意识地敲下z-index:9999,心里清楚这不过是程序...

    动画篇--碎片动画

    本文授权转载,作者:Sindri的小巢(简书)前言从最开始动笔动画篇的博客,至今已经过去了四个多月。这段时间回头看了看自己之前的动画文章,发现用来讲解动画的例子确实不那么的赏心悦目。于是这段时间总是想...

    Nature:大洋转换断层处的拉张构造与两阶段地壳增生

    Nature:大洋转换断层处的拉张构造与两阶段地壳增生转换断层是三种基本的板块边界之一,全球总长度超过48000km(Bird,2003),它们的发现为板块构造理论的建立奠定了重要的基础(Wil...