所用的学习链接:

【奥特学园】ROS机器人入门课程《ROS理论与实践》零基础教程P204-

【以上视频笔记见http://www.autolabor.com.cn/book/ROSTutorials/】

一、目标

产生两只乌龟 A 和 B ,B 会自动运行至A的位置。

键盘只控制 A 的运动,而 B 跟随 A 运动。

乌龟跟随实现的核心,是乌龟 A 和 B 都要发布相对世界坐标系的坐标信息,然后,订阅到该信息需要转换获取A相对于B坐标系的信息,最后,再生成速度信息,并控制B运动。

  1. 启动乌龟显示节点
  2. 在乌龟显示窗体中生成一只新的乌龟(需要使用服务)
  3. 编写两只乌龟发布坐标信息的节点
  4. 编写订阅节点订阅坐标信息并生成新的相对关系生成速度信息

二、准备工作

1.创建环境

① 创建工作空间

# mkdir -p 空间名称/src   # 必须得有 src
mkdir -p 7.24_turtle/src
cd 7.24_turtle
catkin_make# 启动
code .

② 创建功能包 tf1_turtle

右键src ---> create catkin package→自定义package name(例 tf1_turtle)→dependencies:roscpp rospy std_msgs tf2 tf2_ros tf2_geometry_msgs geometry_msgs turtlesim

roscpp rospy std_msgs tf2 tf2_ros tf2_geometry_msgs geometry_msgs turtlesim

如果有包缺失的话可以调用以下代码加载:

# sudo apt-get install ros-<ros版本>-包名
# 包名中的 下划线(_)要换成杠(-)
# sudo apt-get install ros-noetic-包名

ctrl+shift+b 编译运行,选择 catkin_make:build,看是否报错,没报错则依赖包没有出现问题。

③ 创建文件夹

在 tf1_turtle 目录下新建文件夹 launch(存放launch文件)、scripts(存放python文件)

④ 创建 t1_turtle2.cpp 文件

在 src 文件夹下新建文件 t1_turtle2.cpp,用于生成一只新的小乌龟

// 1.头文件
#include "ros/ros.h"
#include "turtlesim/Spawn.h"/*
目标:向服务器发送请求,生成一只新的乌龟话题:/spawn消息:turtlesim/Spawn
1.头文件
2.初始化 ROS 节点
3.创建节点句柄
4.创建客户端对象
5.组织数据并发送
6.处理响应
*/int main(int argc, char *argv[])
{setlocale(LC_ALL,"");// 2.初始化 ROS 节点ros::init(argc,argv,"service_call");// 3.创建节点句柄ros::NodeHandle nh;// 4.创建客户端对象ros::ServiceClient client= nh.serviceClient<turtlesim::Spawn>("/spawn");// 5.组织数据并发送//   5-1.组织请求数据//      创建乌龟turtlesim::Spawn spawn;//      乌龟坐标spawn.request.x=1.0;spawn.request.y=4.0;//      转向 90 度spawn.request.theta=1.57;//      乌龟名字spawn.request.name="turtle2";//   5-2.发送请求//      判断服务器状态//      ros::service::waitForService("/spawn");client.waitForExistence();//      flag 接收响应状态,响应结果也会被设置进 spawn 对象bool flag=client.call(spawn);  // 6.处理响应if(flag){ROS_INFO("乌龟生成成功,新乌龟较:%s",spawn.response.name.c_str());}else{ROS_INFO("请求失败!");}return 0;
}

⑤ 修改 CMakeLists.txt 文件

修改 tf1_turtle 目录下的 CMakeLists.txt 文件

# 原句子
# 1.add_executable(${PROJECT_NAME}_node src/tf1_turtle_node.cpp)
# 2.add_dependencies(${PROJECT_NAME}_node ${${PROJECT_NAME}_EXPORTED_TARGETS} ${catkin_EXPORTED_TARGETS})
# 3. target_link_libraries(${PROJECT_NAME}_node
#   ${catkin_LIBRARIES}
# )# 修改为
# 1.
add_executable(t1_turtle2 src/t1_turtle2.cpp)
# 2.
add_dependencies(t1_turtle2 ${${PROJECT_NAME}_EXPORTED_TARGETS} ${catkin_EXPORTED_TARGETS})
# 3.
target_link_libraries(t1_turtle2${catkin_LIBRARIES}
)

⑥ 创建 t1_turtle1.launch 文件

在 launch 文件夹下新建文件 t1_turtle1.launch

<launch><!-- 启动乌龟GUI节点 --><node pkg="turtlesim" type="turtlesim_node" name="turtle1" output="screen"/><!-- 键盘控制节点 --><!-- 也可以不加这一行,启动后新建终端,运行rosrun turtlesim turtle_teleop_key来启动键盘--><!-- 发布的是关于第一只乌龟的运动消息,因此只能控制第一只 --><node pkg="turtlesim" type="turtle_teleop_key" name="key" output="screen"/><!-- 生成新乌龟的节点 --><node pkg="tf1_turtle" type="t1_turtle2" name="turtle2" output="screen"/>
</launch>

这里我因为 type 写错,出现过 RLException: Unable to launch 的错误,注意检查

 ⑦ 给予文件执行权限

在 7.24_turtle / src / tf1_turtle / src 目录下打开终端,运行

# 使文件成为非root用户也可执行的文件
chmod +x *.cpp

ctrl+shift+b 编译运行,选择 catkin_make:build,看是否报错,没报错则依赖包没有出现问题。

2. 测试运行

roslaunch tf1_turtle t1_turtle1.launch

弹出一个窗口,上面有两只小乌龟。

选中运行 launch 文件的终端窗口为当前窗口,上下左右可以控制小乌龟运动。

三、 消息发布和订阅

(1)新建 t2_pub_turtle.cpp(发布)

#include "ros/ros.h"
#include "tf2_ros/transform_broadcaster.h"
#include "turtlesim/Pose.h"
#include "geometry_msgs/PointStamped.h"
#include "tf2/LinearMath/Quaternion.h"
/*
发布方:订阅乌龟的位姿信息,转换成相对于窗体的坐标系并发布话题:/turtle/pose消息:/turtlesim/Pose1.头文件
2.设置编码、初始化 ros 节点
3.创建订阅对象,订阅 /turtlex/pose
4.回调函数处理订阅的 pose 信息将 pose 信息转换成 TransFormStamped发布
5.spin*/// 声明变量接收传递的参数
std::string turtle_name;void doPose(const turtlesim::Pose::ConstPtr &pose)
{// 获取位姿信息,转换成坐标系相对关系(核心),并发布// a.创建TF发布对象static tf2_ros::TransformBroadcaster pub;// b.组织被发布的数据geometry_msgs::TransformStamped ts;ts.header.frame_id = "world";ts.header.stamp = ros::Time::now();// 关键动态参数// ts.child_frame_id="turtle1";ts.child_frame_id = turtle_name;// 坐标系偏移量设置ts.transform.translation.x = pose->x;ts.transform.translation.y = pose->y;ts.transform.translation.z = 0;// 坐标系四元数/* 位姿信息中没有四元数,但有偏航角度。乌龟为 2D,无翻滚和俯仰角度所以乌龟欧拉角:0 0 theta*/tf2::Quaternion qtn;qtn.setRPY(0, 0, pose->theta);ts.transform.rotation.x = qtn.getX();ts.transform.rotation.y = qtn.getY();ts.transform.rotation.z = qtn.getZ();ts.transform.rotation.w = qtn.getW();// c.发布pub.sendTransform(ts);
}int main(int argc, char *argv[])
{// 2.设置编码、初始化 ROS节点setlocale(LC_ALL, "");ros::init(argc, argv, "dynamic_pub");ros::NodeHandle nh;// 解析通过launch传入的arg参数if (argc != 2){ROS_ERROR("请传入一个参数");return 1;}else{turtle_name = argv[1];}// 3.创建订阅对象,订阅/turtlex/pose// 关键动态参数// ros::Subscriber sub = nh.subscribe("/turtle1/pose",100,doPose);ros::Subscriber sub = nh.subscribe(turtle_name + "/pose", 100, doPose);// 4.回调函数处理订阅的 pose 信息ros::spin();return 0;
}

(2)新建 t3_control_turtle2.cpp(订阅)

#include "ros/ros.h"
#include "tf2_ros/transform_listener.h"
#include "tf2_ros/buffer.h"
#include "geometry_msgs/PointStamped.h"
#include "tf2_geometry_msgs/tf2_geometry_msgs.h"
#include "geometry_msgs/TransformStamped.h"
#include "geometry_msgs/Twist.h"/*
订阅方:计算 turtle1 和 turtle2 的 相对关系计算 角速度和线速度并发布1.头文件
2.编码、初始化、NodeHandle
3.创建订阅对象
4.编写解析逻辑
5.spinOnce()*/int main(int argc, char *argv[])
{// 2.编码、初始化、NodeHandlesetlocale(LC_ALL, "");ros::init(argc, argv, "tfs_sub");ros::NodeHandle nh;// 3.创建订阅对象tf2_ros::Buffer buffer;tf2_ros::TransformListener sub(buffer);// A.创建发布对象ros::Publisher pub = nh.advertise<geometry_msgs::Twist>("/turtle2/cmd_vel", 100);// 4.编写解析逻辑ros::Rate rate(10);while (ros::ok()){// 核心try{// 1.计算 turtle1 和 turtle2 的 相对关系/* 1 相对于 2 的坐标系关系参数1:目标坐标系 B参数2:源坐标系   A参数:ros::Time(0) 取间隔最短的两个坐标系关系帧计算相对关系返回值:geometry_msgs::TransformStamped 源相对于坐标系的相对关系*/geometry_msgs::TransformStamped son1ToSon2 = buffer.lookupTransform("turtle2", "turtle1", ros::Time(0));// ROS_INFO("son1 相对于 son2 的信息:父级%s,子级:%s 偏移量(%.2f,%.2f,%.2f)",//         son1ToSon2.header.frame_id.c_str(),  //turtle2//         son1ToSon2.child_frame_id.c_str(),   //turtle1//         son1ToSon2.transform.translation.x,//         son1ToSon2.transform.translation.y,//         son1ToSon2.transform.translation.z//         );// B.根据相对计算并组织速度消息geometry_msgs::Twist twist;// 线速度,不会横着y,不会飞z,只有x前后// x = 系数 * 开方( y ^ 2 + x ^ 2 )twist.linear.x = 0.5 * sqrt(pow(son1ToSon2.transform.translation.x, 2) + pow(son1ToSon2.transform.translation.y, 2));// 角速度,没有打滚x,没有抬头低头y,只有左右转z// z = 系数 * 反正切( 对边 , 临边 )twist.angular.z = 4 * atan2(son1ToSon2.transform.translation.y, son1ToSon2.transform.translation.x);// C.发布spub.publish(twist);}catch (const std::exception &e){ROS_INFO("错误提示:%s", e.what());}rate.sleep();ros::spinOnce();}// 5.spinOnce()return 0;
}

(3)修改 CMakeLists.txt

修改 tf1_turtle 目录下的 CMakeLists.txt 文件

(4)修改 t1_turtle1.launch 文件

<launch><!-- 1.启动乌龟GUI节点 --><node pkg="turtlesim" type="turtlesim_node" name="turtle1" output="screen" /><!-- 键盘控制节点 --><!-- 也可以不加这一行,启动后新建终端,运行rosrun turtlesim turtle_teleop_key来启动键盘--><!-- 发布的是关于第一只乌龟的运动消息,因此只能控制第一只 --><node pkg="turtlesim" type="turtle_teleop_key" name="key" output="screen" /><!-- 2.生成新乌龟的节点 --><node pkg="tf1_turtle" type="t1_turtle2" name="turtle2" output="screen" /><!-- 3.启动2个乌龟相对于世界的坐标关系的发布因为代码内容相似,因此用一个节点。对该节点启动2次,分别传参 turtle1 和 turtle2 --><node pkg="tf1_turtle" type="t2_pub_turtle" name="pub1" args="turtle1" output="screen" /><node pkg="tf1_turtle" type="t2_pub_turtle" name="pub2" args="turtle2" output="screen" /><!-- 4.订阅坐标信息,转换成 乌龟A 相对于 乌龟B 的坐标信息,再生成控制 乌龟B 的速度信息 --><node pkg="tf1_turtle" type="t3_control_turtle2" name="control" output="screen" /></launch>

四、运行乌龟跟随

roslaunch tf1_turtle t1_turtle1.launch

操控键盘,第二只小乌龟将一直跟随第一只移动。距离越远,第二只小乌龟的加速越大

ROS入门(十)——两只小乌龟(乌龟跟随C++实现)相关推荐

  1. ROS入门五 TF坐标变换

    ROS入门五 TF坐标变换 坐标变换简介 TF功能包 是什么? TF功能包干什么 ? TF坐标变换如何实现? TF工具 乌龟例程中的TF 安装功能包turtle_tf 运行 实现TF的广播和监听功能 ...

  2. ROS入门 小乌龟跟随示例

    一.不使用TF转换的方法 #include <ros/ros.h> #include <geometry_msgs/Twist.h> #include<math.h> ...

  3. ROS入门笔记(十二):动作编程 (C++)

    ROS入门笔记(十二):动作编程 (C++) 文章目录 01 导读 02 功能包的创建 03 在功能包中创建action(动作) 3.1 自定义action 3.2 在package.xml中添加功能 ...

  4. ROS入门笔记(十):编写与测试简单的消息发布器和订阅器(C++)

    ROS入门笔记(十):编写与测试简单的消息发布器和订阅器(C++) 文章目录 01 导读 02 功能包的创建 03 功能包的源代码编写 3.1 编写发布器节点(talker.cpp) 3.2 编写订阅 ...

  5. conan入门(二十):封装只包含头文件(header_only)的库示例

    conan: 封装只包含头文件(header_only)的库示例 有的C/C++项目只包含头文件,不需要编译,对于这种情况如何封装为Conan的包呢? Conan官方文档 <Package sc ...

  6. 古月居 ROS入门21讲 第十二讲 话题消息的定义与使用

    古月居 ROS入门21讲 第十二讲 话题消息的定义与使用 Person.msg string name uint8 sex uint8 ageuint8 unknown=0 uint8 male=1 ...

  7. 诗和远方:无题(五十二)- 写给认识的一个老姐的两只猫

    在网络上里认识了一个老姐,她养了两只猫,分别叫墨颜和金枪,于是我说我给两只猫写一首诗吧: 墨雨挥洒心系画,颜神并茂聚笔尖 与时逝去纸无迹,篱外细看意浮生 金穗稻满秋意浓,黑土深扎草人立 枪指何方谁可知 ...

  8. ROS入门 TF与URDF

    一.什么是TF TF全程就是transform,就是一个坐标系的转换.在ROS中坐标的转换是一个很重要的内容,主要还是因为机器的不灵活性,如果是人,完全可以灵活地控制手臂去抓取一个物体,但是换作机器就 ...

  9. ROS入门跟着我就够了(二)上 ROS通信机制

    由于这一章东西比较多,我分了上下两篇,下部分可以在< ROS 入门跟着我就够了>专辑中查看 ROS 中的基本通信机制主要有如下三种实现策略: 话题通信(发布订阅模式)服务通信(请求响应模式 ...

  10. ROS入门21讲 | ROS机器人入门教程 【简明笔记】

    古月·ROS入门21讲 | 一学就会的ROS机器人入门教程 文章目录 ROS核心概念 ROS命令行 工作空间与功能包 订阅与发布 发布者 Publisher 订阅者 Subscriber 话题消息的自 ...

最新文章

  1. 【组队学习】曹志宾:基于Python的会员数据化运营
  2. Java lambda 循环累加求和
  3. 我的2013年度总结
  4. 曾经用过的Sql Server分页方法小结
  5. linux的编程命令,linux编程常用命令
  6. 4月22日MySQL学习
  7. poj 1230(贪心)
  8. 开启算法编程之旅的准备工作——如何在Windows 上安装 Anaconda 和 PyCharm
  9. poi向word插入图片_如何使用word裁剪图片图形?如何使用word修整图片?
  10. 删除unique key,删除原有的联合主键,新建新的联合逐渐,删除索引,修改索引,mysql5.7修改索引名称,查看最大连接数量,查看是否有ssl
  11. 在 GitHub 上提交代码必备指南!
  12. 巧用ftp命令(一)--利用ftp分发ssh密钥
  13. 41. 后台模块开发(6)
  14. Atitit opencv3.0  3.1 3.2 新特性attilax总结
  15. Java开发银行管理系统
  16. 在制作Windows虚拟机模板时的常用技巧
  17. 目前主要的网络试衣系统
  18. POS58票据热敏打印机,怎么用ESC/POS命令控制打印文字大小?
  19. 普通正态分布如何转换到标准正态分布
  20. HTTP请求415错误 – 不支持的媒体类型(Unsupported media type)

热门文章

  1. CF235C Cyclical Quest(SAM)
  2. Vue.js / Nuxt.js / uni-app - 移动端 H5 网页,在电脑 PC 上打开时居中显示(手机网页在电脑上打开后,简单粗暴防止 “界面样式“ 拉伸变形的解决方案)适用所有web项目
  3. STM32_HAL_SPI_ADS1256调试记录
  4. 使用CA签发的服务器证书搭建Tomcat双向SSL认证服务
  5. keytool生成证书
  6. 国税服务器反回文件错误,电子税务局常见问题解答电子税务局异常转办(一).pdf...
  7. 广东海洋大学微型计算机控制技术,LCD显示实验(综合性)
  8. Nginx 限流的天坑
  9. P1_C1-3:系统分析与设计概要
  10. 密码破解---John the Ripper使用