c++创建订阅者和发布者

2023-10-11 06:32
文章标签 c++ 创建 订阅 发布者

本文主要是介绍c++创建订阅者和发布者,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

此文章默认读者已经对ros有了一定的基础,明白ros之间如何进行数据通信,了解ros的文件结构,了解工作空间,功能包的概念。本篇文章参考创客智造上的教程,他们的讲解比我更详细。百度一下创客智造即可。

  订阅者和发布者依托于节点,即订阅者和发布者是在节点中完成的。所以先来说一下节点的定义,步骤可分为:

1.新建代码源文件

2.写代码

3.在cmakelist.txt文件中定义

订阅者和发布者在写代码的步骤中。

 

下面上一段,roswiki上给出的官方demo,然后对demo做讲解。

talker.cpp

#include "ros/ros.h"

#include "std_msgs/String.h"

 

#include <sstream>  //c++里的导包,对于我这种c++半路出家的,就理解为java里的import语句

//值得一说的是第一行语句,include ros/ros.h  ,引入ros的相关头文件,这是所有ros开发的起点

 

/**

 * This tutorial demonstrates simple sending of messages over the ROS system.

 */

int main(int argc, char **argv)

{

  /**

   * The ros::init() function needs to see argc and argv so that it can perform

   * any ROS arguments and name remapping that were provided at the command line.

   * For programmatic remappings you can use a different version of init() which takes

   * remappings directly, but for most command-line programs, passing argc and argv is

   * the easiest way to do it.  The third argument to init() is the name of the node.

   *

   * You must call one of the versions of ros::init() before using any other

   * part of the ROS system.

   */

  ros::init(argc, argv, "talker");    //初始化节点,第一二个参数走main里传进来,第三个参数为节点名称

 

  /**

   * NodeHandle is the main access point to communications with the ROS system.

   * The first NodeHandle constructed will fully initialize this node, and the last

   * NodeHandle destructed will close down the node.

   */

•   ros::NodeHandle n;   //实例化节点

 

 

  /**

   * The advertise() function is how you tell ROS that you want to

   * publish on a given topic name. This invokes a call to the ROS

   * master node, which keeps a registry of who is publishing and who

   * is subscribing. After this advertise() call is made, the master

   * node will notify anyone who is trying to subscribe to this topic name,

   * and they will in turn negotiate a peer-to-peer connection with this

   * node.  advertise() returns a Publisher object which allows you to

   * publish messages on that topic through a call to publish().  Once

   * all copies of the returned Publisher object are destroyed, the topic

   * will be automatically unadvertised.

   *

   * The second parameter to advertise() is the size of the message queue

   * used for publishing messages.  If messages are published more quickly

   * than we can send them, the number here specifies how many messages to

   * buffer up before throwing some away.

   */

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

//定义一个发布者,尖括号内的泛型是ros里的消息格式,后面的参数分别为话题名称和消息队列长度

  ros::Rate loop_rate(10);

//设置循环的频率

 

 

  /**

   * A count of how many messages we have sent. This is used to create

   * a unique string for each message.

   */

  int count = 0;

  while (ros::ok())  //ros::ok表示ros的状态,当程序出问题或者ctrl+c退出时为false

  {

    /**

     * This is a message object. You stuff it with data, and then publish it.

     */

    std_msgs::String msg;  //定义需要发布的数据

 

    std::stringstream ss;

    ss << "hello world " << count;

    msg.data = ss.str();

 

    ROS_INFO("%s", msg.data.c_str());  //ros_info是ros提供的打印信息的方法

 

    /**

     * The publish() function is how you send messages. The parameter

     * is the message object. The type of this object must agree with the type

     * given as a template parameter to the advertise<>() call, as was done

     * in the constructor above.

     */

    chatter_pub.publish(msg);

 

    ros::spinOnce();  

 

loop_rate.sleep();  //类似于线程里的sleep方法

 

    ++count;

  }

  return 0;

}

 

 

 

Listener.cpp

#include "ros/ros.h"

#include "std_msgs/String.h"

 

/**

 * This tutorial demonstrates simple receipt of messages over the ROS system.

 */

//回调函数,用作数据的处理

void chatterCallback(const std_msgs::String::ConstPtr& msg)

{

  ROS_INFO("I heard: [%s]", msg->data.c_str());

}

 

int main(int argc, char **argv)

{

  /**

   * The ros::init() function needs to see argc and argv so that it can perform

   * any ROS arguments and name remapping that were provided at the command line.

   * For programmatic remappings you can use a different version of init() which takes

   * remappings directly, but for most command-line programs, passing argc and argv is

   * the easiest way to do it.  The third argument to init() is the name of the node.

   *

   * You must call one of the versions of ros::init() before using any other

   * part of the ROS system.

   */

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

 

  /**

   * NodeHandle is the main access point to communications with the ROS system.

   * The first NodeHandle constructed will fully initialize this node, and the last

   * NodeHandle destructed will close down the node.

   */

  ros::NodeHandle n;

 

  /**

   * The subscribe() call is how you tell ROS that you want to receive messages

   * on a given topic.  This invokes a call to the ROS

   * master node, which keeps a registry of who is publishing and who

   * is subscribing.  Messages are passed to a callback function, here

   * called chatterCallback.  subscribe() returns a Subscriber object that you

   * must hold on to until you want to unsubscribe.  When all copies of the Subscriber

   * object go out of scope, this callback will automatically be unsubscribed from

   * this topic.

   *

   * The second parameter to the subscribe() function is the size of the message

   * queue.  If messages are arriving faster than they are being processed, this

   * is the number of messages that will be buffered up before beginning to throw

   * away the oldest ones.

   */

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

//实例化订阅者对象,第一个参数为订阅的话题名称,第二个参数为消息队列长度

   第三个参数为回调函数,即接收到数据后要执行的方法

 

  /**

   * ros::spin() will enter a loop, pumping callbacks.  With this version, all

   * callbacks will be called from within this thread (the main one).  ros::spin()

   * will exit when Ctrl-C is pressed, or the node is shutdown by the master.

   */

  ros::spin();

 

  return 0;

}

 

相比较来说,话题的订阅者比发布者更复杂一点,发布者只关注将数据发布出去,订阅者更关注数据接收到之后如何做处理,当然并不是一个节点只能定义一个发布者或者订阅者,一个发布者发布的数据可以是本话题里订阅处理后的数据,最重要的还是让整个系统条理清晰。 节点和话题的命名最好做到结构清晰,让人望文知义。

代码写好之后,接下来就是编译,并且发布到ros系统中。Ros工程使用cmake构建,所以ros的编译要编写cmakelist.txt,然后使用ros的编译命令catkin_make进行编译。所以如果要进行ros更层次的开发,了解一下cmakelist的语法也是必须的。

首先找到功能包根目录下的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)

 

cmakelist文件编写好之后,要进入工作空间的根目录,进行编译:

在工作空间根目录下运行命令catkin_make,若不报错则成功

 

 

验证一下;

终端1:roscore启动ros

终端2:rosrun 包名(此处为你的包名) talker

终端3:rosrun 包名 listener

观察屏幕打印信息

这篇关于c++创建订阅者和发布者的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/186227

相关文章

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal