ROS整理 —— 话题的发布与订阅(3)

先举个栗子:

  • 乌龟案例的Publisher与Subscriber
    • 运行ROS Master: roscore
    • 运行仿真器节点: rosrun turtlesim turtlesim_node
    • 运行键盘控制的节点: rosrun turtlesim turtle_teleop_key
    • 查看节点关系图: rqt_graph

注:图中,teleop_turtle节点创建Publisher,发布速度命令;turtlesim节点创建Subscriber,订阅速度命令;/turtle1/cmd_vel 是话题。

 

一、创建发布(Publisher)

前提:

/**
 * 该例程将发布chatter话题,消息类型String
 */
 
#include <sstream>                      // 用于字符串和其他数据类型转换
#include "ros/ros.h"                    // 包含了ROS里大部分常用的头文件
#include "std_msgs/String.h"

int main(int argc, char **argv)
{
    // ROS节点初始化:输入值,Publish节点名
    ros::init(argc, argv, "talker");

    // 创建节点句柄
    ros::NodeHandle n;

    // 创建一个Publisher,发布名为chatter的topic,消息类型为std_msgs::String,队列大小
    ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);

    // 设置循环的频率:调用Rate::sleep()时,据此设置休眠时间,此处单位Hz
    ros::Rate loop_rate(10);

    int count = 0;
    while (ros::ok())                           //  roscpp会默认安装一个SIGINT句柄,它负责处理Ctrl-C键盘操作——使得ros::ok()返回FALSE。
    {
        // 初始化std_msgs::String类型的消息
        std_msgs::String msg;                   // 实例化类型
        std::stringstream ss;
        ss << "hello world " << count;          // 将输出数值给了ss
        msg.data = ss.str();                    // 该String消息类型只有一个成员data,将ss和msg相互关联,这样的话后面的输出做准备

        // 发布消息
        ROS_INFO("%s", msg.data.c_str());       // 打印:c_str()把string 对象转换成c中的字符串样式
        chatter_pub.publish(msg);

        // 循环等待回调函数(建议默认加入该函数)
        ros::spinOnce();

        // 按照循环频率延时(休眠时间)
        loop_rate.sleep();
        ++count;
    }

    return 0;
}

/*
ros::ok()返回false,如果下列条件之一发生:
   SIGINT接收到(Ctrl-C);
   被另一同名节点踢出ROS网络;
   ros::shutdown()被程序的另一部分调用;
   所有的ros::NodeHandles都已经被销毁.
一旦ros::ok()返回false, 所有的ROS调用都会失效。
*/
/*
语法: 
const char *c_str();
c_str()函数返回一个指向正规C字符串的指针常量, 内容与本string串相同. 
这是为了与c语言兼容,在c语言中没有string类型,故必须通过string类对象的成员函数c_str()把string 对象转换成c中的字符串样式。
*/

二、创建订阅(Subscriber)

前提:

  • 在learning_communication下建一个名为 listener.cpp 的文件,并将下列代码复制进去;
/**
 * 该例程将订阅chatter话题,消息类型String
 */
 
#include "ros/ros.h"
#include "std_msgs/String.h"

// 接收到订阅的消息后,会进入消息回调函数
void chatterCallback(const std_msgs::String::ConstPtr& msg)
{
    // 将接收到的消息打印出来
    ROS_INFO("I heard: [%s]", msg->data.c_str());
}

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

    // 创建节点句柄
    ros::NodeHandle n;

    // 创建一个Subscriber,订阅名为chatter的topic,注册回调函数chatterCallback
    ros::Subscriber sub = n.subscribe("chatter", 1000, chatterCallback);

    // 循环等待回调函数
    ros::spin();

    return 0;
}

注:

  • C++是一种编译语言,运行前需将代码编译成可执行文件
  • ROS编译器使用的是CMake,编译规则在CMakeLists.txt中;

 

三、编译功能包

前提:

  • 打开learning_communication下的CMakeLists.txt文件,做如下更改:
include_directories(
  include
  ${catkin_INCLUDE_DIRS}
)    //设置头文件的相对路径

add_executable(talker01 src/talker.cpp)        // 设置需要编译的代码和生成可执行文件
target_link_libraries(talker01 ${catkin_LIBRARIES})               // 设置链接库
add_dependencies(talker01 ${PROJECT_NAME}_generate_messages_cpp)  // 设置依赖

add_executable(listener01 src/listener.cpp)                       // 如上
target_link_libraries(listener01 ${catkin_LIBRARIES})
add_dependencies(listener01 ${PROJECT_NAME}_generate_messages_cpp)

执行:

  • cd ~/catkin_ws
  • catkin_make

结果:在 ~/catkin_ws/devel/lib/learning_communication 下会生成两个可执行文件:talker01listener01

 

四、运行Publisher与Subscriber

步骤:

  • 启动管理器节点: roscore
  • 启动发布节点: rosrun learning_communication talker01

  • 启动订阅节点: rosrun learning_communication listener01

注:

  • Publisher 和 Subscriber 节点的启动顺序可以调换,没有要求!!!
  • rosrun的用法:
    • rosrun [package_name] [node_name]
    • 此处的 [node_name]是catkin_make编译后至于devel/lib/<package name> 下的可执行文件名

 

参考网址:

我的ROS入门(五):总算搞通ROS的服务节点订阅发布消息话题了:https://blog.csdn.net/liulj95/article/details/47680599

ROS学习笔记(一)补充篇 参考创客制造:https://blog.csdn.net/david_han008/article/details/54730228

char** argv与char *argv[]区别和strcat()函数:https://blog.csdn.net/weixin_41217046/article/details/79700483

char *a 与char a[] 的区别和char** argv与char *argv[]区别:https://blog.csdn.net/u011068702/article/details/52588455

【ROS学习】(四)ROS消息传递——std_msgs:https://blog.csdn.net/wengge987/article/details/50614957

 

 

  • 1
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ROS1中的话题发布订阅是一种常用的通信机制,允许节点之间通过特定的话题(topics)进行数据交换和信息传递。发布者(publisher)节点负责发布数据到特定话题,而订阅者(subscriber)节点则负责订阅话题并接收发布发布的数据。 在ROS1中,话题是一种基于消息(message)的通信方式。发布者和订阅者通过特定的消息类型进行通信。在给定的话题上进行订阅发布之前,需要先定义该话题消息类型。 例如,我们可以使用ROS消息定义语言(msg)来定义一个名为"example"的消息类型,该消息类型具有两个字段"test1"和"test2"(引用和)。然后,我们可以使用这个消息类型来创建发布者和订阅者节点。 发布者节点的工作是在指定的话题发布特定消息。在示例代码中,pub.cpp(引用)中的发布者节点定义了一个名为"/topic/example"的话题,并将消息类型设置为"example"。通过调用ros::Publisher对象的publish()函数,发布者可以将消息发送到该话题上。 订阅者节点的工作是在指定的话题订阅并接收发布发布消息。在示例代码中,sub.cpp(引用)中的订阅者节点通过调用ros::Subscriber对象的subscribe()函数来订阅名为"topic/example"的话题,并且指定了一个名为subscriberCallback的回调函数来处理接收到的消息。在回调函数中,可以对接收到的消息进行相应的处理。 综上所述,ROS1中的话题发布订阅是通过定义消息类型、创建发布者和订阅者节点,并在特定的话题上进行发布订阅来实现的。发布者负责发布消息话题,而订阅者则负责订阅话题并接收发布发布消息。这种通信机制可以方便地实现节点之间的数据交换和信息传递。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值