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

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

####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 关注

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • Gitea

    Gitea 是一个开源社区驱动的轻量级代码托管解决方案,后端采用 Go 编写,采用 MIT 许可证。

    4 引用 • 16 回帖 • 2 关注
  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 242 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    16 引用 • 53 回帖 • 118 关注
  • Love2D

    Love2D 是一个开源的, 跨平台的 2D 游戏引擎。使用纯 Lua 脚本来进行游戏开发。目前支持的平台有 Windows, Mac OS X, Linux, Android 和 iOS。

    14 引用 • 53 回帖 • 512 关注
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖 • 3 关注
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 401 关注
  • QQ

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

    45 引用 • 557 回帖 • 227 关注
  • Kotlin

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

    19 引用 • 33 回帖 • 25 关注
  • 开源中国

    开源中国是目前中国最大的开源技术社区。传播开源的理念,推广开源项目,为 IT 开发者提供了一个发现、使用、并交流开源技术的平台。目前开源中国社区已收录超过两万款开源软件。

    7 引用 • 86 回帖
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖 • 4 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    51 引用 • 37 回帖
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 683 关注
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    311 引用 • 1666 回帖 • 2 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1083 引用 • 3461 回帖 • 286 关注
  • flomo

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

    3 引用 • 80 回帖 • 1 关注
  • Mac

    Mac 是苹果公司自 1984 年起以“Macintosh”开始开发的个人消费型计算机,如:iMac、Mac mini、Macbook Air、Macbook Pro、Macbook、Mac Pro 等计算机。

    164 引用 • 594 回帖 • 1 关注
  • 大数据

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

    89 引用 • 113 回帖 • 1 关注
  • FFmpeg

    FFmpeg 是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。

    22 引用 • 31 回帖 • 3 关注
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 549 关注
  • 钉钉

    钉钉,专为中国企业打造的免费沟通协同多端平台, 阿里巴巴出品。

    15 引用 • 67 回帖 • 370 关注
  • Oracle

    Oracle(甲骨文)公司,全称甲骨文股份有限公司(甲骨文软件系统有限公司),是全球最大的企业级软件公司,总部位于美国加利福尼亚州的红木滩。1989 年正式进入中国市场。2013 年,甲骨文已超越 IBM,成为继 Microsoft 后全球第二大软件公司。

    103 引用 • 126 回帖 • 452 关注
  • DNSPod

    DNSPod 建立于 2006 年 3 月份,是一款免费智能 DNS 产品。 DNSPod 可以为同时有电信、网通、教育网服务器的网站提供智能的解析,让电信用户访问电信的服务器,网通的用户访问网通的服务器,教育网的用户访问教育网的服务器,达到互联互通的效果。

    6 引用 • 26 回帖 • 521 关注
  • 链书

    链书(Chainbook)是 B3log 开源社区提供的区块链纸质书交易平台,通过 B3T 实现共享激励与价值链。可将你的闲置书籍上架到链书,我们共同构建这个全新的交易平台,让闲置书籍继续发挥它的价值。

    链书社

    链书目前已经下线,也许以后还有计划重制上线。

    14 引用 • 257 回帖
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    108 引用 • 54 回帖 • 1 关注
  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 457 关注
  • 星云链

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

    3 引用 • 16 回帖 • 5 关注