ROS2入门教程—创建一个简单的订阅者和发布者(C++版)

  • 1 创建功能包
  • 2 创建发布者节点
  • 3 设置发布者节点依赖项
  • 4 设置发布者节点编译规则
  • 5 创建订阅者
  • 6 编译并运行

  节点是通过ROS graph进行通信的可执行进程。在本教程中,节点将通过话题以字符串消息的形式相互传递信息。这里使用的例子是一个简单的“talker”和“listener”系统;一个节点发布数据,另一个节点订阅该话题,以便它可以接收该数据。之前我们已经通过命令行实现过话题的发布和订阅,本篇我们就来尝试下如何通过C++代码来实现发布者和订阅者。

1 创建功能包

  首先打开一个新终端,并且设置环境变量,以便ros2命令能够正常工作。然后进入到dev_ws/src文件夹,运行创建功能包的指令:

ros2 pkg create --build-type ament_cmake cpp_pubsub

  您的终端将返回一条消息,验证功能包cpp_pubsub及其所有必要文件和文件夹的创建。

going to create a new package
package name: cpp_pubsub
destination directory: /home/libo/dev_ws/src
package format: 3
version: 0.0.0
description: TODO: Package description
maintainer: ['libo <libo@todo.todo>']
licenses: ['TODO: License declaration']
build type: ament_cmake
dependencies: []
creating folder ./cpp_pubsub
creating ./cpp_pubsub/package.xml
creating source and include folder
creating folder ./cpp_pubsub/src
creating folder ./cpp_pubsub/include/cpp_pubsub
creating ./cpp_pubsub/CMakeLists.txt

2 创建发布者节点

  在cpp_pubsub功能包的src文件夹下,创建一个发布者节点的代码文件publisher_member_function.cpp,然后拷贝以下代码放进去:

#include <chrono>
#include <functional>
#include <memory>
#include <string>#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"using namespace std::chrono_literals;/* This example creates a subclass of Node and uses std::bind() to register a* member function as a callback from the timer. */class MinimalPublisher : public rclcpp::Node
{public:MinimalPublisher(): Node("minimal_publisher"), count_(0){publisher_ = this->create_publisher<std_msgs::msg::String>("topic", 10);timer_ = this->create_wall_timer(500ms, std::bind(&MinimalPublisher::timer_callback, this));}private:void timer_callback(){auto message = std_msgs::msg::String();message.data = "Hello, world! " + std::to_string(count_++);RCLCPP_INFO(this->get_logger(), "Publishing: '%s'", message.data.c_str());publisher_->publish(message);}rclcpp::TimerBase::SharedPtr timer_;rclcpp::Publisher<std_msgs::msg::String>::SharedPtr publisher_;size_t count_;};int main(int argc, char * argv[]){rclcpp::init(argc, argv);rclcpp::spin(std::make_shared<MinimalPublisher>());rclcpp::shutdown();return 0;}

  接下来对上述代码进行简单的注释。代码的顶部包括您将要使用的标准C++头文件。rclcpp/rclcpp.hpp是ROS2中常用C++接口的头文件,使用C++编写的ROS2节点程序一定需要包含该头文件。std_msgs/msg/string.hpp是ROS2中字符串消息的头文件,后边我们会周期发布一个HelloWorld的字符串消息,所以需要包含该头文件。

#include <chrono>
#include <functional>
#include <memory>
#include <string>#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"using namespace std::chrono_literals;

  下一行通过继承rclcpp::Node节点基类创建节点类MinimalPublisher。代码中的每个this都引用节点。

class MinimalPublisher : public rclcpp::Node

  接下来是节点类MinimalPublisherd的构造函数,将count_变量初始化为0,节点名初始化为“minimal_publisher”。构造函数内先是创建了一个发布者,发布的话题名是topic,话题消息是String,保存消息的队列长度是10,然后创建了一个定时器timer_,做了一个500ms的定时,每次触发定时器后,都会运行回调函数timer_callback

public:MinimalPublisher(): Node("minimal_publisher"), count_(0){publisher_ = this->create_publisher<std_msgs::msg::String>("topic", 10);timer_ = this->create_wall_timer(500ms, std::bind(&MinimalPublisher::timer_callback, this));}

  timer_callback是这里的关键,每次触发都会发布一次话题消息。message中保存的字符串是Hello world加一个计数值,然后通过RCLCPP_INFO宏函数打印一次日志信息,再通过发布者的publish方法将消息发布出去。

Private:void timer_callback(){auto message = std_msgs::msg::String();message.data = "Hello, world! " + std::to_string(count_++);RCLCPP_INFO(this->get_logger(), "Publishing: '%s'", message.data.c_str());publisher_->publish(message);}

  最后是计时器、发布者和计数器字段的声明。

rclcpp::TimerBase::SharedPtr timer_;
rclcpp::Publisher<std_msgs::msg::String>::SharedPtr publisher_;
size_t count_;

  定义完节点类后还需要编写main函数,先初始化ROS2节点,然后使用rclcpp::spin创建MinimalPublisher,并且进入自旋锁,当退出锁时,就会关闭节点结束了。完成以上发布者的代码后,功能包里还有一些内容需要设置。

int main(int argc, char * argv[])
{rclcpp::init(argc, argv);rclcpp::spin(std::make_shared<MinimalPublisher>());rclcpp::shutdown();return 0;
}

3 设置发布者节点依赖项

  打开功能包的package.xml文件,根据前面的教程,先把<description>, <maintainer><license>这些基础信息填写好:

<description>Examples of minimal publisher/subscriber using rclcpp</description>
<maintainer email="you@email.com">Your Name</maintainer>
<license>Apache License 2.0</license>

  然后还需要添加依赖项,放到ament_cmake下边:

<buildtool_depend>rclcpp</buildtool_depend>
<buildtool_depend>std_msgs</buildtool_depend>

4 设置发布者节点编译规则

  接下来打开CMakeLists.txt文件,在find_package语句下,新加入两行:

find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)

  然后再设置具体的编译规则,:

add_executable(talker src/publisher_member_function.cpp)
ament_target_dependencies(talker rclcpp std_msgs)

  最后还要设置安装的规则,这样ros2 run命令才找得到可执行文件:

install(TARGETStalkerDESTINATION lib/${PROJECT_NAME})

  完整的CMakeLists.txt文件应该就是这样的:

cmake_minimum_required(VERSION 3.5)
project(cpp_pubsub)Default to C++14
if(NOT CMAKE_CXX_STANDARD)set(CMAKE_CXX_STANDARD 14)
endif()if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")add_compile_options(-Wall -Wextra -Wpedantic)
endif()find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)add_executable(talker src/publisher_member_function.cpp)
ament_target_dependencies(talker rclcpp std_msgs)install(TARGETStalkerDESTINATION lib/${PROJECT_NAME})ament_package()

5 创建订阅者

  回到dev_ws/src/cpp_pubsub/src文件夹下,创建订阅者节点的代码subscriber_member_function.cpp

#include <memory>#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"
using std::placeholders::_1;class MinimalSubscriber : public rclcpp::Node
{public:MinimalSubscriber(): Node("minimal_subscriber"){subscription_ = this->create_subscription<std_msgs::msg::String>("topic", 10, std::bind(&MinimalSubscriber::topic_callback, this, _1));}private:void topic_callback(const std_msgs::msg::String::SharedPtr msg) const{RCLCPP_INFO(this->get_logger(), "I heard: '%s'", msg->data.c_str());}rclcpp::Subscription<std_msgs::msg::String>::SharedPtr subscription_;
};int main(int argc, char * argv[])
{rclcpp::init(argc, argv);rclcpp::spin(std::make_shared<MinimalSubscriber>());rclcpp::shutdown();return 0;
}

  订阅者的代码整体流程和发布者类似,现在的节点名叫minimal_subscriber,构造函数中创建了订阅者,订阅String消息,订阅的话题名叫做“topic”,保存消息的队列长度是10,当订阅到数据时,会进入回调函数topic_callback。根据前面的教程,发布者和订阅者使用的话题名称和消息类型必须匹配才能进行通信。

public:MinimalSubscriber(): Node("minimal_subscriber"){subscription_ = this->create_subscription<std_msgs::msg::String>("topic", 10, std::bind(&MinimalSubscriber::topic_callback, this, _1));}

  回调函数中会收到String消息,然后并没有做太多处理,只是通过RCLCPP_INFO打印出来。

private:void topic_callback(const std_msgs::msg::String::SharedPtr msg) const{RCLCPP_INFO(this->get_logger(), "I heard: '%s'", msg->data.c_str());}rclcpp::Subscription<std_msgs::msg::String>::SharedPtr subscription_;

  main函数中的内容和发布者几乎是一致的,就不再赘述。它们的唯一区别就在于:对于发布者节点而言,rclcpp::spin意味着启动计时器,但对于订阅者而言,rclcpp::spin仅仅意味着随时准备接收消息。

  由于该节点的依赖项和发布者一样,我们就不需要修改package.xml文件了,不过编译规则CMakeList.txt还是得加一些内容:

add_executable(listener src/subscriber_member_function.cpp)
ament_target_dependencies(listener rclcpp std_msgs)install(TARGETStalkerlistenerDESTINATION lib/${PROJECT_NAME})

6 编译并运行

  编译前先确认下功能包的依赖项有没有都安装好,在dev_ws路径下运行如下命令:

rosdep install -i --from-path src --rosdistro eloquent -y

  安装完毕后还是在该路径下编译cpp_pubsub功能包:

colcon build --packages-select cpp_pubsub

  编译完成后,打开一个新的终端,设置工作空间的环境变量后,运行发布者:

. install/setup.bash
ros2 run cpp_pubsub talker

  运行成功后可以看到终端每隔0.5s打印一次日志信息:

[INFO] [minimal_publisher]: Publishing: "Hello World: 0"
[INFO] [minimal_publisher]: Publishing: "Hello World: 1"
[INFO] [minimal_publisher]: Publishing: "Hello World: 2"
[INFO] [minimal_publisher]: Publishing: "Hello World: 3"
[INFO] [minimal_publisher]: Publishing: "Hello World: 4"

  再打开一个新的终端,设置工作空间的环境变量后,运行订阅者:

. install/setup.bash
ros2 run cpp_pubsub listener

  订阅者启动后,终端中会显示当前订阅者收到的消息内容:

[INFO] [minimal_subscriber]: I heard: "Hello World: 10"
[INFO] [minimal_subscriber]: I heard: "Hello World: 11"
[INFO] [minimal_subscriber]: I heard: "Hello World: 12"
[INFO] [minimal_subscriber]: I heard: "Hello World: 13"
[INFO] [minimal_subscriber]: I heard: "Hello World: 14"

  在终端中按Ctrl+C即可退出。到此为止,本篇我们一起创建了两个节点,并且通过话题实现了两个节点之间的通信。

ROS2入门教程—创建一个简单的订阅者和发布者(C++版)相关推荐

  1. ROS2入门教程—创建ROS2功能包(C++版)

    ROS2入门教程-创建ROS2功能包(C++版) 1 ROS2中的功能包 2 创建功能包 3 编译功能包 4 设置环境变量 5 运行功能包 6 功能包中的内容 7 修改package.xml文件    ...

  2. node.js入门 - 2.创建一个简单聊天室

    这篇文章将通过开发一个简单聊天室的方式,介绍node.js的net模块. 一.第一版,只向客户端发送信息   我们先实现一个简单的版本,代码如下: var net=require('net'); va ...

  3. c语言标准函数库怎么建立教程,C语言入门教程-创建一个函数库

    描述 创建一个函数库 上述程序中的rand和bubble_sort函数很实用,很可能在您写其他程序时也能派上用场.为了能更方便地重复使用,您可以为它们创建一个实用工具函数库. 所有的函数库都包括两部分 ...

  4. php 对象教程,创建一个简单的PHP对象_PHP教程

    name = "亚古兽"; $agu -> hitPoint = 50; $agu -> attack = "12"; $agu -> def ...

  5. Qt图形界面编程入门(创建一个简单的程序)

    1,手工编码方式 利用手工编码方式建立"Hello Qt!"程序 第一步: 得到界面 2,无UI的向导方式 从图中,我们发现向导为窗口程序提供了3个基类,分别外QMainWindo ...

  6. D3入门教程——做一个简单的图表

    文章目录 一.准备 1.什么是画布 1)SVG 2)Canvas 3)两者之前的区别 二.开始 1.添加画布 2.绘制矩形 三.完整代码 一.准备 1.什么是画布 HTML 5 提供两种强有力的&qu ...

  7. [转]Express入门教程:一个简单的博客

    来源:http://www.open-open.com/lib/view/open1454560780730.html 转载于:https://www.cnblogs.com/hcbin/p/5317 ...

  8. WF4.0入门系列1——创建一个简单的工作流

    WF4.0入门系列1--创建一个简单的工作流 打开VS2010,选择文件-新建-项目,选择Workflow项 工作流台应用程序,在名称处输入chapter01,选择合适的位置,这里默认,单击确定. V ...

  9. java qq ui界面_java swing 创建一个简单的QQ界面教程

    记录自己用java swing做的第一个简易界面. LoginAction.java package com.QQUI0819; import javax.swing.*; import java.a ...

最新文章

  1. python中的元类_python中的元类
  2. getContext(),getApplicationContext(),getBaseContext()和“ this”之间的区别
  3. Python的subprocess子进程和管道进行交互
  4. MySQL高级 - SQL优化 - limit优化
  5. 是学习Java还是Python?一张图告诉你!
  6. CF819E:Mister B and Flight to the Moon(构造、归纳法)
  7. 《数据结构C语言版》——栈和队列详解(图文并茂),从零开始的学习
  8. js中避免函数名和变量名跟别人冲突
  9. Oracle JDE 系统架构总结..
  10. Android6.0源码下载
  11. 地图客户端自动化测试
  12. 错误: 此上下文中不允许函数定义。
  13. Typhoon-v1.02渗透笔记
  14. Designing Specification
  15. html表格列表模板,前端基础 - HTML(二) 表格、表单、列表
  16. 计算机控制电缆灰色和蓝色,计算机电缆、控制电缆区别
  17. 在nuxt中使用sass
  18. 【Linux】grep命令与正则表达式(RegExp)
  19. 从参数到使用体验,家用电视机选购攻略奉上
  20. matlab中伽马函数的使用

热门文章

  1. Telegram桌面端(tdesktop)编译方法
  2. 主流的深度学习模型有哪些?
  3. 分子数据的获取、解析与结构绘制(RDKit)
  4. 不可忘却的纪念:又10部经典日剧(转)
  5. docker<应用分享> 发布镜像到阿里云、从阿里云拉取镜像
  6. 穷举算法——鸡兔同笼问题
  7. Docker系列(二十四)——Docker实例六Docker安装Redis实例
  8. mac上超级好用的划词、截图翻译器bob
  9. 图书管理后台简单实现(三)
  10. rstp 小米网络摄像头_常见网络摄像机的端口及RTSP地址