ROS源代码之Publish底层实现(一)

@[TOC]ROS源代码之Publish底层实现ROS源代码之Publish底层实现在之前对ROS的源码的学习中,基本弄清楚了ROS的topic通信方式中,节点发布/订阅的机制和原理,可以说解释了节点与master之间的交流方式。但是对于节点与节点之间通信的具体过程,却一笔带过,所以这次通过再次阅读这部分源码,来明确节点在advertise之后,注册完成之后,publish消息时底层通信的逻辑和形式。本次源码阅读主要通过dfs的方式来进行。首先是ROS wiki提供的发布者节点发布信息时的最表层调用:
摘要由CSDN通过智能技术生成

@[TOC]ROS源代码之Publish底层实现

ROS源代码之Publish底层实现

在之前对ROS的源码的学习中,基本弄清楚了ROS的topic通信方式中,节点发布/订阅的机制和原理,可以说解释了节点与master之间的交流方式。但是对于节点与节点之间通信的具体过程,却一笔带过,所以这次通过再次阅读这部分源码,来明确节点在advertise之后,注册完成之后,publish消息时底层通信的逻辑和形式。
本次源码阅读主要通过dfs的方式来进行。首先是ROS wiki提供的发布者节点发布信息时的最表层调用:

#include "ros/ros.h"
#include "std_msgs/String.h"
#include <sstream>

int main(int argc, char **argv)
{
    ros::init(argc, argv, "talker");
  
    ros::NodeHandle n;
    
    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());
        chatter_pub.publish(msg);
        ros::spinOnce();
        loop_rate.sleep();        
    }

    return 0;
}

可以看到,其中调用publish函数的正是之前我们使用nodehandle对象的advertise注册发布者时所返回的publisher类对象chatter_pub,传入的参数时我们自定义的msg类型。
我们在roscpp包中寻找相关的代码,如下是在publisher.h中所定义的两种重载的模板函数,分别用来应传入参数类型不同的的两种情况。

······
//共享指针类型的引用传参
    template <typename M>
      void publish(const boost::shared_ptr<M>& message) const
    {
      using namespace serialization;

      if (!impl_)
        {
          ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher");
          return;
        }

      if (!impl_->isValid())
        {
          ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher (topic [%s])", impl_->topic_.c_str());
          return;
        }

      ROS_ASSERT_MSG(impl_->md5sum_ == "*" || std::string(mt::md5sum<M>(*message)) == "*" || impl_->md5sum_ == mt::md5sum<M>(*message),
                     "Trying to publish message of type [%s/%s] on a publisher with type [%s/%s]",
                     mt::datatype<M>(*message), mt::md5sum<M>(*message),
                     impl_->datatype_.c_str(), impl_->md5sum_.c_str());

      SerializedMessage m;
      m.type_info = &typeid(M);
      m.message = message;

      // 这里使用ref函数是给bind绑定的参数传入ref()引用包装类型,使参数为引用传递;
      // 使用bind应该是为了把message绑定给serializeMessage
      publish(boost::bind(serializeMessage<M>, boost::ref(*message)), m);
    }
// M类型的引用传参 泛型
    template <typename M>
      void publish(const M& message) const
    {
      using namespace serialization;
      namespace mt = ros::message_traits;

      if (!impl_)
        {
          ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher");
          return;
        }

      if (!impl_->isValid())
        {
          ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher (topic [%s])", impl_->topic_.c_str());
          return;
        }

      ROS_ASSERT_MSG(impl_->md5sum_ == "*" || std::string(mt::md5sum<M>(message)) == "*" || impl_->md5sum_ == mt::md5sum<M>(message),
                     "Trying to publish message of type [%s/%s] on a publisher with type [%s/%s]",
                     mt::datatype<M>(message), mt::md5sum<M>(message),
                     impl_->datatype_.c_str(), impl_->md5sum_.c_str());

      SerializedMessage m;
      publish(boost::bind(serializeMessage<M>, boost::ref(message)), m);
    }
······

他们最终都调用的publish函数在publisher.cpp中实现,具体如下:

    ········
// publish函数
void Publisher::publish(const boost::function<SerializedMessage(void)>& serfunc, SerializedMessage& m) const
{
    // impl是否存在
  if (!impl_)
  {
    ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher (topic [%s])", impl_->topic_.c_str());
    return;
  }
  // impl是否可用
  if (!impl_->isValid())
  {
    ROS_ASSERT_MSG(false, "Call to publish() on an invalid Publisher (topic [%s])", impl_->topic_.c_str());
    return;
  }
  // topicManager的单例调用publish
  TopicManager::instance()->publish(impl_->topic_, serfunc, m);
}
    ···········

其中的impl是publisher类在传参构造时的生成一个内部类对象,一般是由nodehandle类中的advertise函数,获取到ops的相关信息并传进来生成的impl_对象。也就是说在注册成功后才会存在的一个对象。
显然之后,Publisher.publish()函数让TopicManager的唯一实例调用了

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: ROS (Robot Operating System) 是一个非常强大的机器人操作系统,它支持多种平台和硬件。在实现 ROS STM32 无人船底层之前,需要先了解ROS是如何工作的以及ROS的基本概念。 ROS是一个分布式的系统,它由多个节点组成,并且这些节点可以在不同的计算机上运行。每个节点可以发布和订阅消息,以实现节点之间的通信。节点之间的通信是通过ROS消息传递机制来实现的。 对于STM32无人船底层实现,可以使用ROS的serial通信协议,将STM32与ROS进行连接。在STM32中,需要实现ROS的串口通信协议,并且实现ROS消息数据的解析和封装。具体的步骤如下: 1. 实现ROS的串口通信协议 ROS的串口通信协议是一种基于ASCII字符的通信协议,用于在ROS节点之间进行通信。在STM32中,可以使用UART串口来实现ROS的串口通信协议。需要实现的功能包括: - 发送数据:将STM32的数据封装成ROS消息,并通过UART串口发送给ROS节点。 - 接收数据:从ROS节点接收ROS消息,并解析成STM32可以处理的数据。 2. 实现ROS消息数据的解析和封装 在STM32中,需要实现ROS消息数据的解析和封装。ROS消息是一种结构化的数据类型,它由多个字段组成,每个字段具有自己的数据类型和名称。在STM32中,需要实现ROS消息数据的解析和封装,以便能够将ROS消息转换为STM32可以处理的数据类型。 3. 实现STM32驱动程序 在STM32中,需要实现底层驱动程序,以便能够控制无人船的各个部分。需要实现的功能包括: - 控制电机:通过PWM信号控制电机的转速。 - 控制舵机:通过PWM信号控制舵机的角度。 - 读取传感器数据:通过ADC模块读取传感器数据。 4. 在ROS中创建节点 在ROS中,需要创建一个节点来与STM32进行通信。该节点将接收STM32发送的数据,并将数据封装成ROS消息。同时,该节点还将向STM32发送控制指令,以控制无人船的运动。 5. 测试与调试 最后,需要对实现的STM32无人船底层进行测试和调试。可以使用ROS的调试工具,如rqt和rostopic,来查看ROS消息的发送和接收情况。同时,也可以使用STM32的调试工具,如ST-Link和Keil uVision,来查看STM32的运行情况。 总体来说,实现ROS STM32无人船底层需要具备ROS和STM32的基本知识,同时需要熟悉ROS消息传递机制和STM32的硬件驱动程序。 ### 回答2: ROS(机器人操作系统)是一个灵活且强大的开源软件平台,用于构建机器人系统。STM32是一款常用的嵌入式微控制器系列,具有良好的性能和可靠性。无人船是一种通过自主导航来进行航行任务的无人驾驶船只。 在无人船的底层实现中,ROS可以提供以下功能和特性: 1. 操作系统:ROS提供一个功能齐全的操作系统,包括任务调度、进程管理、内存管理和设备驱动程序等,可以为无人船提供可靠的底层支持。 2. 通信机制:ROS提供了灵活的通信机制,可以实现无人船与各种传感器、执行器和其他设备的数据交换和命令传递。例如,可以通过ROS中的消息传递机制来获取传感器数据,并通过ROS服务或话题发布器向执行器发送控制命令。 3. 硬件驱动:ROS具有完善的硬件驱动程序支持,可以与STM32微控制器进行通信和交互。通过ROS底层硬件驱动接口,可以实现与STM32的通信、数据传输和控制功能。 4. 导航和建图:ROS提供了先进的导航和建图功能包,可以帮助无人船实现定位、路径规划和避障等任务。通过使用ROS导航功能包,可以实现无人船的自主导航能力。 5. 仿真和调试:ROS提供了强大的仿真和调试工具,可以对无人船的行为进行模拟和调试,以验证算法和系统设计的正确性和性能。 在底层实现中,STM32可以作为无人船的控制器,负责接收和处理来自传感器的数据,并根据ROS提供的指令执行相应的控制动作。通过ROS和STM32的结合,可以实现无人船的智能控制、感知和决策能力,使其能够根据环境变化进行自主导航、目标追踪和路径规划等任务。 ### 回答3: ROS(机器人操作系统)是一种开源的机器人操作系统,提供了一套通用的软件框架和工具,使机器人的开发和控制更加简单和高效。而STM32是一款由STMicroelectronics公司开发的低功耗微控制器,广泛应用于嵌入式系统的开发中。 在无人船的底层实现中,ROS和STM32可以很好地结合应用。首先,在ROS中可以使用ROS的硬件驱动库(如roscpp、rosserial等)来连接STM32与电脑或其他设备进行通信。其次,通过ROS的话题(topic)和服务(service)机制,可以将STM32的传感器数据和执行指令与其他ROS节点进行交互。 对于无人船的底层实现,STM32可以用来控制和驱动各种船只所需的硬件设备,比如电机、舵机、传感器等。通过STM32的GPIO口、PWM输出等功能,可以将底层硬件与ROS节点进行连接和控制。 例如,无人船底层实现中的传感器数据可以通过STM32采集,并通过ROS的话题机制发布到相应的ROS节点。而无人船的控制指令(如速度、转向等)可以通过ROS的服务机制发送给STM32进行执行。 此外,如果需要实现一些高级的算法和决策,如路径规划、SLAM(同时定位与地图构建)等,可以通过ROS中的各种现有包和工具来实现。而STM32作为底层的硬件平台,则负责执行这些算法和决策生成的指令,控制无人船的运动。 综上所述,ROS和STM32的结合可以实现无人船的底层控制和硬件驱动,通过ROS的通信机制和算法库,实现与其他节点的数据交互和高级算法的应用。这样的结合使得底层实现更加灵活和通用,为无人船系统的开发和控制提供了便捷的解决方案。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值