创建简单的发布者和订阅者

创建发布者和订阅者

发布者

  • 切换到软件包src目录,创建一个发布节点
roscd <package_name>
cd src
touch talker.cpp
  • 发布节点talker.cpp的内容如下
#include <ros/ros.h> // 导入ROS系统包含核心公共文件
#include <std_msgs/String.h> // 导入std_msgs/String消息头文件,这个由std_msgs包的string.msg自动生成
#include <sstream>

/**
 * 发布方实现:
 * 1.包含头文件
 * 2.初始化ROS节点
 * 3.创建节点句柄
 * 4.创建发布者对象
 * 5.编写发布逻辑并发布数据
*/

int main(int argc, char **argv)
{
  ros::init(argc, argv, "talker"); // ROS节点初始化,指定节点名称为"talker",节点名称要保持唯一性

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

  // 创建一个Publisher,发布名为chatter的topic,消息类型为std_msgs::String,队列长度1000 
  // 第二个参数为消息队列长度,队列中最大保存的消息数量, 超过这个数量, 旧的消息会被丢弃(先进先销毁)
  ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);
  
  // 创建一个循环频率对象, 指定发布消息的频率
  ros::Rate loop_rate(10);

  int count = 0;
  
  ros::Duration(3).sleep(); // 等待3s, 以便发布节点注册到ROS Master

  while (ros::ok()) // 检查节点是否应继续运行,当接收到SIGINT(Ctrl+C)或者 ros::shutdown() 被调用时返回false
  {
    ++count;
    std_msgs::String msg; // 初始化std_msgs::String msg类型的消息

    std::stringstream ss; // 初始化std::stringstream类型的消息
    ss << "hello world --- " << count; // 定义字符串流"hello world"并赋给ss
    msg.data = ss.str(); // 最后转换为字符串赋值给 msg.data
    ROS_INFO_STREAM(" " << msg->data.c_str()); // 输出调试信息

    chatter_pub.publish(msg); // 发布消息

    ros::spinOnce(); // 处理传入消息的回调函数, 不是必需要的,但是保持增加这个调用,是好习惯

    loop_rate.sleep(); // 按照循环频率延时
  }
  return 0;
}

总结

  • 初始化ROS系统

  • 向ROSMaster注册节点信息,包括发布的话题名和话题中的消息类型

  • 创建消息数据

  • 按照一定的频率循环发布消息

订阅者

  • 同样
roscd <package_name>
cd src
touch listener.cpp
  • 订阅节点listener.cpp的内容如下
#include <ros/ros.h> // 导入ROS系统包含核心公共头文件
#include <std_msgs/String.h> // 导入std_msgs/String消息头文件,这个是由std_msgs包的string.msg文件自动生成

/**
 * 订阅方实现:
 * 1.包含头文件
 * 2.初始化ROS节点
 * 3.创建节点句柄
 * 4.创建订阅者对象
 * 5.编写回调函数处理接收到的数据
 * 6.设置循环调用回调函数
*/

// 接收到chatter话题的消息后,回调函数就会被调用并且收到的消息会作为参数
void chatterCallback(const std_msgs::String::ConstPtr& msg)
{
  ROS_INFO_STREAM("I heard: " << msg->data.c_str()); // 将接收到的消息打印出来
}

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

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

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

  ros::spin(); // 启动自循环,等待回调函数

  return 0;
}

总结

  • 初始化ROS节点
  • 订阅需要的话题
  • 循环等待话题消息,接收到信息后进入回调函数
  • 在回调函数中完成消息处理

修改CMakeLists.txt

  • find_package依赖项
# 将message_generation加在括号闭合前
find_package(catkin REQUIRED COMPONENTS
        roscpp
        rospy
        std_msgs
        message_generation
)
  • 调用生成的消息
generate_messages(
        DEPENDENCIES
        std_msgs
)
  • 设置生成可执行文件和链接库(添加到CMakeLists.txt文件的底部)
add_executable(talker src/talker.cpp)
target_link_libraries(talker ${catkin_LIBRARIES})

add_executable(listener src/listener.cpp)
target_link_libraries(listener ${catkin_LIBRARIES})

句柄

  • 句柄(Handle)这个概念可以理解为一个“把手”,你握住了门把手,就可以很容易把整扇门拉开,而不必关心门是什么样子。NodeHandle就是对节点资源的描述,有了它你就可以操作这个节点了,比如为程序提供服务、监听某个topic上的消息、访问和修改param等等

  • NodeHandle是Node的句柄,用来对当前节点进行各种操作。在ROS中,NodeHandle是一个定义好的类,通过include<ros/ros.h>,我们可以创建这个类,以及使用它的成员函数

NodeHandle

publisher

  • 用句柄创建publisher
ros::Publisher publisher = NodeHandle.advertise<message_type>(topic, queue_size);
  • 编写消息内容
  • 发布消息
publisher.publish(message);

subscriber

  • 通过调用节点句柄的 subscriber() 方法来订阅话题
ros::Subscriber subscriber = NodeHandle.subscriber(topic, queue_size, callblack_function);
  • 持续订阅话题,直到你取消订阅

参数

  • 使用C++来获取参数
nodeHandle.getParam(param_name, variable)
  • 当函数找到对应的参数就会返回true,否则返回false

四种节点句柄命名空间的种类

  • ros::NodeHandle:默认命名空间
  • ros::NodeHandle("~"):私有命名空间
  • ros::NodeHandle("ksals"):指定命名空间
  • ros::NodeHandle("/"):全局命名空间(不推荐)

这些节点的命名空间将解析为

  • /namespace/topic
  • /namespace/node/topic
  • /namespace/ksals/topic
  • /topic
posted @ 2021-11-11 21:57  TNTksals  阅读(129)  评论(0编辑  收藏  举报