[ROS]编写简单的发布器和订阅器

本贴最后更新于 3006 天前,其中的信息可能已经沧海桑田

####1.编写发布器节点

节点是 ROS 中连接 ROS 网络的可执行程序的专业术语。这里我们创建一个发布器节点 talker,它将连续的广播消息。

首先进入 beginner_tutorials 包:

cd ~/catkin_ws/src/beginner_tutorials

#####1.1 代码

在 beginner_tutorials 包目录下创建 src 目录:

mkdir -p ~/catkin_ws/src/beginner_tutorials/src

src 目录包含了 beginner_toturials 包的所有的源文件。
在 src 目录下创建 talker.cpp 文件,添加一下内容:

#include "ros/ros.h"
#include "std_msgs/String.h"

#include <sstream>

/**
 * 
 *本教程演示简单的发送消息到ROS系统
 */
int main(int argc, char **argv)
{
  /**
   * ros::init()方法需要argc和argv参数,这样就可以接收任意在命令行提供的ROS参数和名字。
   * 对于程序化的映射,你可以使用不同版本的init()来直接进行映射,
   *
   * 在使用ROS系统的其他部分之前,你必须调用一个版本的ros::init()。
   */
  ros::init(argc, argv, "talker");

  /**
   * NodeHandle是与ROS系统交流的main入口。
   * 第一个NodeHandle构造器将完全的初始化节点,最后一个NodeHandle析构器将关闭节点。
   */
  ros::NodeHandle n;

  /**
   * advertise()方法提供方式告诉ROS在给定名字的主题上发布(消息)。
   * 这将产生一个对ROSmaster节点的调用,ROS的master节点保存了发布器和订阅器的注册。
   * 在advertise()执行完后,master节点将通知每一个向这个主题订阅的订阅器,订阅成功之后,它们将与主题所在的节点进行点对点的通信。
   * advertise()返回一个发布器对象,这个发布器对象允许通过主题调用publish()方法发布消息。一旦发布器的所有副本都销毁了,这个主题将不会再自动的发布消息。

   * advertise()方法的第二个参数是用来发布消息的队列的大小。如果发布消息比发送消息的速度快,这个参数的作用就是指定了缓存的消息数目。
   */
  ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);

  ros::Rate loop_rate(10);

  /**
   * 已经发送的消息的数目. 这个用来为每一个消息创建一个唯一的字符串
   */
  int count = 0;
  while (ros::ok())
  {
    /**
     * 消息对象. 赋值之后发布.
     */
    std_msgs::String msg;

    std::stringstream ss;
    ss << "hello world " << count;
    msg.data = ss.str();

    ROS_INFO("%s", msg.data.c_str());

    /**
     * publish()方法用来发送消息。参数是消息对象。对象的类型必须符合上面advertise<>()构造器中申明的<>中的参数类型。
     */
    chatter_pub.publish(msg);

    ros::spinOnce();

    loop_rate.sleep();
    ++count;
  }


  return 0;
}

#####1.2.代码解析

接下来分析上面的代码。

#include "ros/ros.h"

ros/ros.h 头文件包含了 ROS 系统的大部分公共类库。

#include "std_msgs/String.h"

上面的头文件包含了 std_msgs 包中的 std_msgs/String 消息。这个头文件自动地根据 std_msgs 包中的 String.msg 文件生成。更多的信息参考 http://wiki.ros.org/msg。

ros::init(argc, argv, "talker");

初始化 ROS。这句话允许 ROS 从命令行完成名字的映射,同时这句代码也指定了节点的名字。节点的名字在系统运行的时候必须唯一,并且取名字的时候名字必须符合基本的取名规则,比如说名字就不能有/。

 ros::NodeHandle n;

创建节点的 handle。首次创建的 NodeHandle 将完成节点的初始化,最后的一次析构将节点使用的任何资源。

ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);

向 master 节点声明我们将要通过 chatter 主题发布 std_msgs/String 类型的数据。master 节点将通知所有监听 chatter 的节点 chatter 主题将要发布数据。第二个参数是发布队列的大小。如果发布的消息太快,超过了 1000 的最大缓存消息之后将会开始丢失旧的消息。
NodeHandle::advertise()方法返回 ros::Publisher 对象,这个方法提供了两种用途:
1)使用 publis()方法通过创建的主题发布消息;
2)一旦超过了范围将自动的停止发布。

ros::Rate loop_rate(10);

ros::Rate 对象允许你指定具体的发布频率。它将追踪距离上一次调用 Rate::sleep()已经过去多久,保证正确的休眠时间。这里我们指定它的速率是 10HZ。

  int count = 0;
  while (ros::ok())
  {

默认的,roscpp 将安装一个监听 ctrl+c 组合键的 SIGINT handler,按下组合键之后将导致 ros::ok()返回 false。
下面情况将导致 ros::ok()返回 false。
’‘’
1.SIGINT handler 接收到组合键 ctrl+c;
2.被另外一个重名节点踢出网络;
3.程序其他部分调用了 ros::shutdown();
4.所有的 ros::NodeHandles 都被销毁。
‘’‘
一旦 ros::ok()返回 false,所有的 ROS 调用将失败。

 std_msgs::String msg;

    std::stringstream ss;
    ss \<\\\< \"hello world \" \<\< count;
    msg.data = ss.str();

在 ROS 上发布的消息是使用消息适用类,这个类是根据 msg 文件生成的。复杂的数据类型也是可能的,但是现在,我们只使用标准的 String 消息,这个消息只有一个成员:”data“。

chatter_pub.publish(msg);

现在我们可以向任意连接了的节点广播消息了。

    ROS_INFO("%s", msg.data.c_str());

ROS_INFO 以及类似的代替了 printf/cout。更多的参考:http://wiki.ros.org/rosconsole

ros::spinOnce();

调用 ros::spinOnce()在简单编程的时候是没必要的,因为我们没有接收任何的回调。然而,如果我们需要在程序中添加订阅器,但是有没有在这里调用 ros::spinOnce(),回调将不会执行。所以这里最好将这句加上。

loop_rate.sleep();

上面这句话让 ros::Rate 对象休眠一会儿,以保证我们是以 10HZ 的速率发布消息。

让我们总结下之前的工作:

1.初始化ROS系统;
2.向master节点注册:之后将通过chatter主题发布std_msgs/String消息;
3.以一秒为周期,循环发布十次消息。

现在我们继续编写一个节点来接收消息。

####2.编写订阅器节点

#####2.1 源码

在 beginner_tutorials 包下的 src 下创建 listener.cpp 文件,并粘贴下面的内容:

#include "ros/ros.h"
#include "std_msgs/String.h"

/**
 * 本教程编写了简单的从ROS系统接收消息的消息接收器。
 */
void chatterCallback(const std_msgs::String::ConstPtr& msg)
{
  ROS_INFO("I heard: [%s]", msg->data.c_str());
}

int main(int argc, char **argv)
{
  /**
   * 初始化节点,名称为listener
   */
  ros::init(argc, argv, "listener");


  ros::NodeHandle n;

  ros::Subscriber sub = n.subscribe("chatter", 1000, chatterCallback);

  /**
   *循环处理回调,所有的回调都是主线程中调用。按ctrl+c退出或者节点呗master关闭
   */
  ros::spin();

  return 0;
}

2.2 代码解释

接下来一点点的解析代码,这里只解析之前没有的:

void chatterCallback(const std_msgs::String::ConstPtr& msg)
{
  ROS_INFO("I heard: [%s]", msg->data.c_str());
}

这个回调方法将会被调用,当有新的消息到达 chatter 主题。消息是通过一种 http://www.boost.org/doc/libs/1_37_0/libs/smart_ptr/shared_ptr.htm 的方法。这意味着你可以保存消息,而不用担心需要你去删除,不用复制底层数据。

有很多中方式处理回调的实现。roscpp_tutorlals 包有一些例子。参考:http://wiki.ros.org/roscpp/Overview。

总结:

1.初始化ROS系统
2.订阅chatter主题;
3.等待消息;
4.当消息达到,chatterCallback()方法将被调用。

3.构建节点

通过之前的教程,包初始化会自动生成 package.xml 和 CMakeList.txt 文件。

在 CMakeList.txt 低端加上:

include_directories(include ${catkin_INCLUDE_DIRS})

add_executable(talker src/talker.cpp)
target_link_libraries(talker ${catkin_LIBRARIES})
add_dependencies(talker beginner_tutorials_generate_messages_cpp)

add_executable(listener src/listener.cpp)
target_link_libraries(listener ${catkin_LIBRARIES})
add_dependencies(listener beginner_tutorials_generate_messages_cpp)

上面的代码会创建 2 个可执行程序,talker 和 listener,默认的位置是 ~/catkin_ws/devel/lib/<package_name>.

之后执行:catkin_make。

关于依赖和构建安装程序,参考:http://wiki.ros.org/catkin/CMakeLists.txt

编译完成之后运行:

rosrun beginner_tutorials talker 

可以看到:

[INFO] [WallTime: 1314931831.774057] hello world 1314931831.77
[INFO] [WallTime: 1314931832.775497] hello world 1314931832.77
[INFO] [WallTime: 1314931833.778937] hello world 1314931833.78
[INFO] [WallTime: 1314931834.782059] hello world 1314931834.78
[INFO] [WallTime: 1314931835.784853] hello world 1314931835.78
[INFO] [WallTime: 1314931836.788106] hello world 1314931836.79

运行:


rosrun beginner_tutorials listener 

可以看到:

[INFO] [WallTime: 1314931969.258941] /listener_17657_1314931968795I heard hello world 1314931969.26
[INFO] [WallTime: 1314931970.262246] /listener_17657_1314931968795I heard hello world 1314931970.26
[INFO] [WallTime: 1314931971.266348] /listener_17657_1314931968795I heard hello world 1314931971.26
[INFO] [WallTime: 1314931972.270429] /listener_17657_1314931968795I heard hello world 1314931972.27
[INFO] [WallTime: 1314931973.274382] /listener_17657_1314931968795I heard hello world 1314931973.27
[INFO] [WallTime: 1314931974.277694] /listener_17657_1314931968795I heard hello world 1314931974.28
[INFO] [WallTime: 1314931975.283708] /listener_17657_1314931968795I heard hello world 1314931975.28

这样就完成了简单的发布器和订阅器的编写了。

  • ROS
    23 引用 • 2 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Quicker

    Quicker 您的指尖工具箱!操作更少,收获更多!

    32 引用 • 130 回帖 • 2 关注
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 27 关注
  • AngularJS

    AngularJS 诞生于 2009 年,由 Misko Hevery 等人创建,后为 Google 所收购。是一款优秀的前端 JS 框架,已经被用于 Google 的多款产品当中。AngularJS 有着诸多特性,最为核心的是:MVC、模块化、自动化双向数据绑定、语义化标签、依赖注入等。2.0 版本后已经改名为 Angular。

    12 引用 • 50 回帖 • 474 关注
  • QQ

    1999 年 2 月腾讯正式推出“腾讯 QQ”,在线用户由 1999 年的 2 人(马化腾和张志东)到现在已经发展到上亿用户了,在线人数超过一亿,是目前使用最广泛的聊天软件之一。

    45 引用 • 557 回帖 • 67 关注
  • Pipe

    Pipe 是一款小而美的开源博客平台。Pipe 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    132 引用 • 1114 回帖 • 124 关注
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖 • 4 关注
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    8 引用 • 26 回帖
  • Eclipse

    Eclipse 是一个开放源代码的、基于 Java 的可扩展开发平台。就其本身而言,它只是一个框架和一组服务,用于通过插件组件构建开发环境。

    75 引用 • 258 回帖 • 617 关注
  • Logseq

    Logseq 是一个隐私优先、开源的知识库工具。

    Logseq is a joyful, open-source outliner that works on top of local plain-text Markdown and Org-mode files. Use it to write, organize and share your thoughts, keep your to-do list, and build your own digital garden.

    6 引用 • 63 回帖
  • OpenStack

    OpenStack 是一个云操作系统,通过数据中心可控制大型的计算、存储、网络等资源池。所有的管理通过前端界面管理员就可以完成,同样也可以通过 Web 接口让最终用户部署资源。

    10 引用 • 4 关注
  • 大数据

    大数据(big data)是指无法在一定时间范围内用常规软件工具进行捕捉、管理和处理的数据集合,是需要新处理模式才能具有更强的决策力、洞察发现力和流程优化能力的海量、高增长率和多样化的信息资产。

    93 引用 • 113 回帖
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    62 引用 • 289 回帖 • 1 关注
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖
  • 前端

    前端技术一般分为前端设计和前端开发,前端设计可以理解为网站的视觉设计,前端开发则是网站的前台代码实现,包括 HTML、CSS 以及 JavaScript 等。

    247 引用 • 1348 回帖
  • 小薇

    小薇是一个用 Java 写的 QQ 聊天机器人 Web 服务,可以用于社群互动。

    由于 Smart QQ 从 2019 年 1 月 1 日起停止服务,所以该项目也已经停止维护了!

    34 引用 • 467 回帖 • 742 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 63 关注
  • SEO

    发布对别人有帮助的原创内容是最好的 SEO 方式。

    35 引用 • 200 回帖 • 22 关注
  • 服务器

    服务器,也称伺服器,是提供计算服务的设备。由于服务器需要响应服务请求,并进行处理,因此一般来说服务器应具备承担服务并且保障服务的能力。

    125 引用 • 588 回帖
  • abitmean

    有点意思就行了

    29 关注
  • VirtualBox

    VirtualBox 是一款开源虚拟机软件,最早由德国 Innotek 公司开发,由 Sun Microsystems 公司出品的软件,使用 Qt 编写,在 Sun 被 Oracle 收购后正式更名成 Oracle VM VirtualBox。

    10 引用 • 2 回帖 • 6 关注
  • RabbitMQ

    RabbitMQ 是一个开源的 AMQP 实现,服务器端用 Erlang 语言编写,支持多种语言客户端,如:Python、Ruby、.NET、Java、C、PHP、ActionScript 等。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

    49 引用 • 60 回帖 • 362 关注
  • CloudFoundry

    Cloud Foundry 是 VMware 推出的业界第一个开源 PaaS 云平台,它支持多种框架、语言、运行时环境、云平台及应用服务,使开发人员能够在几秒钟内进行应用程序的部署和扩展,无需担心任何基础架构的问题。

    5 引用 • 18 回帖 • 167 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖
  • etcd

    etcd 是一个分布式、高可用的 key-value 数据存储,专门用于在分布式系统中保存关键数据。

    5 引用 • 26 回帖 • 528 关注
  • flomo

    flomo 是新一代 「卡片笔记」 ,专注在碎片化时代,促进你的记录,帮你积累更多知识资产。

    5 引用 • 107 回帖
  • C

    C 语言是一门通用计算机编程语言,应用广泛。C 语言的设计目标是提供一种能以简易的方式编译、处理低级存储器、产生少量的机器码以及不需要任何运行环境支持便能运行的编程语言。

    85 引用 • 165 回帖 • 1 关注
  • Openfire

    Openfire 是开源的、基于可拓展通讯和表示协议 (XMPP)、采用 Java 编程语言开发的实时协作服务器。Openfire 的效率很高,单台服务器可支持上万并发用户。

    6 引用 • 7 回帖 • 94 关注