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

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

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

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

    相关推荐

    Opinion丨Struggle Against U.S. Mind colonization in the Global South

    Editor'snote:Thismonth,XinhuaNewsAgency'sThinkTankreleasedareporttitled"Colonizationof...

    爱可可AI论文推介(2020.11.4)_爱可可女装旗舰店

    LG-机器学习CV-计算机视觉CL-计算与语言AS-音频与语音RO-机器人(*表示值得重点关注)1、[LG]*CombiningLabelPropagationan...

    何新:罗马伪史考英文版序言_罗马史学

    2019-10-2514:48:27何新:罗马伪史考序言(英文译本)HeXin:PreambleofResearchonPseudo-historyofRome1Afewyear...

    XPeng Stock Rises Over 4% after Q2 Revenue and EV Margin Set Records

    TMTPOST--TheAmericandepositaryreceipts(ADRs)ofXPengInc.rosearound4.2%onTuesdayaftert...

    英汉世界语部首(八)_英文部首字典

    本节讲八个部首,分别是:弓gōng【ECWLrad】bow廾gǒng【ECWLrad】twen广guǎng【ECWLrad】vast己jǐ【ECWLrad】self已yǐ...

    一课译词:划水_划水是什么地方的方言

    [Photo/SIPA]懒惰是人类的天性,因此才总有人会在工作时“划水”。“划水【huáshuǐ】”,本意是指“用胳膊划的动作(makestrokeswithone’sarms)”,延伸为“...

    首测!GPT-4o做Code Review可行吗?

    编辑|言征出品|51CTO技术栈(微信号:blog51cto)近日,OpenAI一记重拳,推出了GPT-4o(“o”表示“omni”),将语音识别和对话方面的优势展示的淋漓尽致。几乎可以肯定,...

    C++|漫谈STL细节及内部原理_c++ stl详解

    1988年,AlexanderStepanov开始进入惠普的PaloAlto实验室工作,在随后的4年中,他从事的是有关磁盘驱动器方面的工作。直到1992年,由于参加并主持了实验室主任BillWo...

    C++ inline关键字深度解析:不止于优化的头文件定义许可

    在C++开发中,几乎每个程序员都用过inline关键字,但多数人只停留在“内联优化”的表层理解。事实上,inline的真正威力在于它打破了C++的单一定义规则(ODR)限制,成为头文件中安全定义函数的...

    实用 | 10分钟教你搭建一个嵌入式web服务器

    之前分享的文章中提到了几种可以在嵌入式中使用的web服务器。嵌入式web服务器就是把web服务器移植到嵌入式系统的服务器。它仍然是基于http文本协议进行通信的,具有标准的接口形式,对客户端...

    中间语言格式_中间格式文本是什么

    在通常情况下,编译器会将目标语言转换成某种中间语言格式,而不是直接将源代码转换成二进制机器指令,不少c语言编译器,都会将代码编译成汇编语言,然后再通过汇编语言编译器将汇编代码转换成目标机器可执行的二进...

    一线开发大牛带你深度解析探讨模板解释器,解释器的生成

    解释器生成解释器的机器代码片段都是在TemplateInterpreterGenerator::generate_all()中生成的,下面将分小节详细展示该函数的具体细节,以及解释器某个组件的机器代码...

    干货,Web开发和前端开发逆天工具大全

    微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(点击页底“阅读原文”前往下载)●●●逆天工具CDN资源库国内Bootstrap中文网开源项目免费CDN服务36...

    移动端rem+vw适配_移动端web页面适配方案

    rem:rem是相对单位,设置根元素html的font-size,比如给html设置字体大小为100px,1rem=100px;rem缺点:1.和根元素font-size值强耦合,系统字...

    从零搭建 React 开发 H5 模板_react html5

    项目创建创建项目文件夹mkdir react-democd react-demonpm init -y依赖安装yarn add rea...