1、发布者程序

在本节中,我们将看到如何发送随机生成的速度指令到一个turtlesim海龟,使它漫无目的地巡游。这个程序的源文件称为pubvel,这个程序展示了从代码中发布消息涉及的所有要素。

其代码如下:

pubvel和hello程序主要的区别都是由于发布消息的需求导致的。

包含消息类型声明     每一个 ROS 话题都与一个消息类型相关联,每一个消息类型都有一个相对应 C++头文件。你需要在你的程序中为每一个用到的消息类型包含这个头文件,代码如下所示:

#include <package_name/type_name.h>

这里应该要注意的是,头文件中的功能包名是定义消息类型的包的名称,而不一定是你自己的包的名称。

在 pubvel 程序中,我们想发布一条类型为 geometry_msgs/Twist 的消息 (名为 geometry_msgs 的包所拥有的类型为 Twist 的消息) ,我们应该这样:

#include <geometry_msgs/Twist.h>

这个头文件的目的是定义一个 C++类,此类和给定的消息类型含有相同的数据类型成员。

这个类定义在以包命名的域名空间中。

这样命名的实际影响是当引用 C++代码中的消息类时,你将会使用双分号(::)来区分开包名和类型名,双分号也称为范围解析运算符。

在我们的 pubvel 例程中,头文件定义了一个名为geometry_msgs::Twist 的类。

创建发布者对象     发布消息的实际工作是由类名为ros::Publisher的一个对象来完成的。类似下面这行的代码创建了我们需要的对象:

ros::Publisher pub = node_handle.advertise<message_type>( topic_name, queue_size);

解析:

---node_handle 是 ros::NodeHandle 类的一个对象,是你在程序的开始处创建的。我们将调用这个对象的 advertise 方法。

---在尖括号中的 message_type 部分,其正式名称为模板参数,是我们要发布的消息的数据类型。这个应该是上面讨论过的头 文 件 中 定 义 的 类 名 。 在 例 程 中 , 我 们 使 用geometry_msgs::Twist 类。

---topic_name 是一个字符串,它包含了我们想发布的话题的名称。 它应该和 rostopic list 或者 rqt_graph 中展示的话题名称一致,但通常没有前斜杠(/) 。我们丢掉前斜杠使话题名为一个相对名称;第 5 章解释了相对名称的机制和目的。在此例程中,话题名为turtle1/cmd_vel。

---advertise 最后的参数是一个整数,表示这个发布者发布的消息序列的大小。在大多数情况下,一个相对比较大的值,比
方说 1000,是合适的。

如果你想从同一个节点发布关于多个话题的消息,你需要为每个话题创建一个独立的 ros::Publisher 对象。

创建并填充消息对象     当我们创建 ros::Publisher 对象时已经引用了消息类。对于消息类型的每个域,这个类都有一个可公共访问的数据成员。

当然, 大多数消息类型含有float64 类型之外的域。 幸运的是,从ROS域类型到C++数据类型的映射正如你所期望的方式工作。一个可能未被注意到的事实是,数组类型(在rosmsg中用方括号表示)在C++代码中是通过STL向量实现的。

发布消息    在所有的前期工作完成后,使用 ros::Publisher 对象的publish 方法可以很简单地发布消息。例如下面所示:
    pub.publish(msg);

这个方法将所给的消息添加到发布者的输出消息队列中,从这里,它会尽快被发送到相同话题的订阅者那里。

定义输出格式    表3.4中的ROS_INFO_STREAM行尽管和发布速度命令不是直接相关的,但还是值得一看的。这是关于宏ROS_INFO_STREAM 可以做什么的一个更加完整的例证, 因为它演示了在输出中除了插入字符串还可以插入其他数据的能力。

2、消息发布循环

程序在这个循环中使用了两个附加的构造函数。

节点是否停止工作的检查     pubvel 的 while 循环的条件是: ros::ok()

---你可能对节点使用了 rosnode kill 命令。

---你可能给程序发送了一个终止信号(Ctrl-C)。

---你可能在程序的某个位置调用了 ros::shutdown() ,这个函数是在你的代码中发送节点工作已经完成信号的一个很有用的方法。

---你可能以相同的名字启动了其他节点,经常是因为你启动了一个相同程序的新实例。

控制消息发布频率     pubvel的最后一个新知识点是它使用了ros::Rate对象:

ros::Rate rate(2);

这个对象控制循环运行速度,其构造函数中的参数以赫兹(Hz)为单位,即每秒钟的循环数。这个例子创建了旨在规范每秒钟执行两个迭代循环的速率对象。邻近每次循环迭代的结尾,我们调用此对象的 sleep 方法: rate.sleep();

你可以使用 rostopic hz 命令来确认这种调控是否工作正常。
对于 pubvel 程序,输出结果类似于:
average rate: 2.000

min: 0.500s max: 0.500s std dev: 0.00006s window: 10

我们看到消息以每秒 2 条的速率发布,且时间上偏差非常小。

3、编译pubvel

适当修改 CMakeLists.txt 和 package.xml, 然后用 catkin_make 来编译你的工作区。然而,它和 hello 例程有一个很重要的区别。

声明消息类型依赖库     因为pubvel使用了来自geometry_msgs包的消息类型,我们必须声明对这个包的依赖关系,这和 3.2.2 节中讨论的 roscpp 依赖库的形式相同。具体而言,除了 roscpp,我们 必 须 修 改 CMakeLists.txt 文 件 的 find_package 行 来 声 明
geometry_msgs: find_package(catkin REQUIRED COMPONENTS roscpp geometry_msgs)

注意: 我们是修改已有的 find_package 行, 而不是新建一行。

在 package.xml 文件中,我们添加新的依赖项:
<build_depend>geometry_msgs</build_depend>
<run_depend>geometry_msgs</run_depend>

别忘了,还有跟hello.cpp程序里类似的修改,一样要添加进去。

回到工作空间编译catkin_make

4、执行 pubvel

第3章 编写ROS程序-2的更多相关文章

  1. 第3章 编写ROS程序-1

    1.创建工作区和功能包 在我们写任何程序之前,第一步是创建一个容纳我们的功能包的工作区,然后再创建功能包本身. 创建工作区  使用标准的mkdir命令行去创建一个目录,我们将把这个新的目录称作工作区目 ...

  2. 第3章 编写ROS程序-3

    1.订阅者程序 我们继续使用 turtlesim 作为测试平台,订阅 turtlesim_node发布的/turtle1/pose 话题. 这一话题的消息描述了海龟的位姿 (位置和朝向) .尽管目前你 ...

  3. C#入门到精通系列课程——第2章编写C#程序

    ◆本章内容 (1)熟悉Visual Studio 2017开发环境 (2)编写第一个C#程序 (3)C#程序结构预览 (4)程序编写规范 (5)难点解答 ◆本章简述 要学习C#编程,必然要熟悉C#程序 ...

  4. 编写ROS程序--HelloROS

    <机器人操作系统浅析>ch3学习记录 A Gentle Introduction to ROS 3.1 创建工作区(工作空间)和功能包 创建工作空间 ~/ros 我们称之为工作区目录 在工 ...

  5. SLAM+语音机器人DIY系列:(二)ROS入门——4.如何编写ROS的第一个程序hello_world

    摘要 ROS机器人操作系统在机器人应用领域很流行,依托代码开源和模块间协作等特性,给机器人开发者带来了很大的方便.我们的机器人“miiboo”中的大部分程序也采用ROS进行开发,所以本文就重点对ROS ...

  6. 第 3 章 编写 PAM 应用程序和服务

    Solaris 开发者安全性指南 Previous: 第 2 章 开发特权应用程序 Next: 第 4 章 编写使用 GSS-API 的应用程序 第 3 章 编写 PAM 应用程序和服务 可插拔验证模 ...

  7. ROS Learning-013 beginner_Tutorials (编程) 编写ROS服务版的Hello World程序(Python版)

    ROS Indigo beginner_Tutorials-12 编写ROS服务版的Hello World程序(Python版) 我使用的虚拟机软件:VMware Workstation 11 使用的 ...

  8. ROS Learning-011 beginner_Tutorials (编程) 编写 ROS 话题版的 Hello World 程序(Python版)

    ROS Indigo beginner_Tutorials-10 编写 ROS 话题版的 Hello World 程序(Python版) 我使用的虚拟机软件:VMware Workstation 11 ...

  9. [Hadoop in Action] 第4章 编写MapReduce基础程序

    基于hadoop的专利数据处理示例 MapReduce程序框架 用于计数统计的MapReduce基础程序 支持用脚本语言编写MapReduce程序的hadoop流式API 用于提升性能的Combine ...

随机推荐

  1. EasyPlayer Android RTSP播放器延迟再优化策略

    EasyPlayer延迟再优化策略 EasyPlayer是一款专门针对RTSP协议进行过优化的播放器.其中两个我们引以为傲的的优点就是起播快和低延迟.最近我们遇到一些需求,其对延迟要求非常苛刻,于是我 ...

  2. opencv使用记录

    /*2017-1-14*/ /*视频的读取...*/ int g_n=0; void on_change(int pos,void *)//看来void*不能省! { printf("g_n ...

  3. Coin和Token有什么区别

    在币圈,经常可以听到“coin”和“token”这些词汇,他们究竟分别代表什么,有什么区别呢?下面本文就和大家一起来扒一扒. 什么是coin? coin (包括山寨coin)是一种数字货币,它通过加密 ...

  4. AndroidUI组件之ImageSwitcher

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/gc_gongchao/article/details/25594669 今天继续AndroidUI组 ...

  5. ABAP screen

    Instance One : SELECTION-SCREEN BEGIN OF BLOCK B1 WITH FRAME TITLE TEXT-100. SELECTION-SCREEN BEGIN ...

  6. CUDA: 共享内存与同步

    CUDA C支持共享内存, 将CUDA C关键字__shared__添加到变量声明中,将使这个变量驻留在共享内存中.对在GPU上启动的每个线程块,CUDA C编译器都将创建该变量的一个副本.线程块中的 ...

  7. 05-树8 File Transfer(25 point(s)) 【并查集】

    05-树8 File Transfer(25 point(s)) We have a network of computers and a list of bi-directional connect ...

  8. 【转】BigInteger、BigDecimal详解

    参考 http://lavasoft.blog.51cto.com/62575/228705/ 从Java4到Java5,Java对BigInteger.BigDecimal两个类功能一直再做扩展与改 ...

  9. CSS3 3D下拉折叠菜单

    在线演示 本地下载

  10. poj The Settlers of Catan( 求图中的最长路 小数据量 暴力dfs搜索(递归回溯))

    The Settlers of Catan Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 1123   Accepted: ...