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

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

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

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • Hprose

    Hprose 是一款先进的轻量级、跨语言、跨平台、无侵入式、高性能动态远程对象调用引擎库。它不仅简单易用,而且功能强大。你无需专门学习,只需看上几眼,就能用它轻松构建分布式应用系统。

    9 引用 • 17 回帖 • 598 关注
  • sts
    2 引用 • 2 回帖 • 148 关注
  • C

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

    83 引用 • 165 回帖 • 44 关注
  • SVN

    SVN 是 Subversion 的简称,是一个开放源代码的版本控制系统,相较于 RCS、CVS,它采用了分支管理系统,它的设计目标就是取代 CVS。

    29 引用 • 98 回帖 • 692 关注
  • Caddy

    Caddy 是一款默认自动启用 HTTPS 的 HTTP/2 Web 服务器。

    10 引用 • 54 回帖 • 127 关注
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    96 引用 • 155 回帖
  • 招聘

    哪里都缺人,哪里都不缺人。

    189 引用 • 1056 回帖 • 2 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 594 关注
  • ReactiveX

    ReactiveX 是一个专注于异步编程与控制可观察数据(或者事件)流的 API。它组合了观察者模式,迭代器模式和函数式编程的优秀思想。

    1 引用 • 2 回帖 • 126 关注
  • Vditor

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

    312 引用 • 1666 回帖 • 1 关注
  • 单点登录

    单点登录(Single Sign On)是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。

    9 引用 • 25 回帖 • 3 关注
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 8 关注
  • 开源

    Open Source, Open Mind, Open Sight, Open Future!

    396 引用 • 3416 回帖
  • Bootstrap

    Bootstrap 是 Twitter 推出的一个用于前端开发的开源工具包。它由 Twitter 的设计师 Mark Otto 和 Jacob Thornton 合作开发,是一个 CSS / HTML 框架。

    18 引用 • 33 回帖 • 683 关注
  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 178 关注
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    164 引用 • 407 回帖 • 526 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

    4 引用 • 7 回帖 • 3 关注
  • Android

    Android 是一种以 Linux 为基础的开放源码操作系统,主要使用于便携设备。2005 年由 Google 收购注资,并拉拢多家制造商组成开放手机联盟开发改良,逐渐扩展到到平板电脑及其他领域上。

    333 引用 • 323 回帖 • 67 关注
  • CodeMirror
    1 引用 • 2 回帖 • 116 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    10 引用 • 85 回帖 • 1 关注
  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 43 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 2 关注
  • GAE

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

    14 引用 • 42 回帖 • 684 关注
  • Vue.js

    Vue.js(读音 /vju ː/,类似于 view)是一个构建数据驱动的 Web 界面库。Vue.js 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件。

    261 引用 • 662 回帖
  • 链滴

    链滴是一个记录生活的地方。

    记录生活,连接点滴

    131 引用 • 3639 回帖
  • OnlyOffice
    4 引用 • 26 关注
  • 笔记

    好记性不如烂笔头。

    303 引用 • 777 回帖