ROS 2边学边练(25)-- 将多个节点组合到一个进程

2024-04-16 10:28

本文主要是介绍ROS 2边学边练(25)-- 将多个节点组合到一个进程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言     

        在ROS 2中,将多个节点(Nodes)组合到一个单独的进程(Process)中通常指的是使用“Composable Nodes”的特性。这个特性允许你定义可复用的组件(Components),然后将这些组件加载到一个或多个ROS 2节点中,而这些节点可以运行在同一个进程中。

这样做有几个好处:

  1. 资源优化:通过将多个节点组合到一个进程中,可以减少进程间通信的开销,并更有效地利用系统资源。

  2. 简化部署:对于需要多个节点协同工作的应用程序,将它们组合到一个进程中可以简化部署和配置。

  3. 代码重用:组件化的设计使得代码重用变得更加容易,因为你可以在不同的节点或应用程序中重用相同的组件。

在ROS 2中实现这一功能,通常涉及以下步骤:

  1. 定义组件:使用ROS 2的组件API定义可复用的组件。每个组件都封装了特定的功能,并且可以在运行时加载到节点中。

  2. 创建节点:创建一个或多个ROS 2节点,这些节点将作为组件的容器。这些节点可以配置为接受组件的加载。

  3. 加载组件:使用ROS 2的组件加载器(Component Loader)将组件加载到节点中。这通常涉及指定组件的类型和参数,并告诉ROS 2在何处找到这些组件的实现。

  4. 运行节点:启动包含组件的节点,这些节点现在将在一个共同的进程中运行,并共享相同的资源。

        需要注意的是,尽管将多个节点组合到一个进程中可以提高效率和简化部署,但这也可能引入额外的复杂性。例如,需要确保进程内的所有节点都能够正确地处理并发操作和资源共享。因此,在决定是否将节点组合到一个进程时,应该仔细考虑应用程序的具体需求和约束。

        简而言之,ROS 2中的“Composable Nodes”特性提供了一种灵活的方式来构建和管理ROS 2应用程序,通过组件化和进程内组合的方式,可以实现更高效、可重用和可配置的机器人和自动化系统。

动动手

运行例子

        这些例子使用了rclcpp_components、ros2component和composition包中的可执行文件,并且可以使用以下命令运行。

发现可用的组件

        想要查看当前工作空间中已注册且能用的组件列表,我们可以利用下面的命令进行查看:

$ros2 component types

返回符合条件的组件列表:

mike@mike-virtual-machine:~/Desktop/ros2_ws$ ros2 component types
examples_rclcpp_wait_setTalkerListener
examples_rclcpp_minimal_subscriberWaitSetSubscriberStaticWaitSetSubscriberTimeTriggeredWaitSetSubscriber
custom_action_cppcustom_action_cpp::FibonacciActionServercustom_action_cpp::FibonacciActionClient
logging_demologging_demo::LoggerConfiglogging_demo::LoggerUsage
compositioncomposition::Talkercomposition::Listenercomposition::NodeLikeListenercomposition::Servercomposition::Client
demo_nodes_cpp_nativedemo_nodes_cpp_native::Talker
robot_state_publisherrobot_state_publisher::RobotStatePublisher
quality_of_service_demo_cppquality_of_service_demo::MessageLostListenerquality_of_service_demo::MessageLostTalkerquality_of_service_demo::QosOverridesListenerquality_of_service_demo::QosOverridesTalker
teleop_twist_joyteleop_twist_joy::TeleopTwistJoy
demo_nodes_cppdemo_nodes_cpp::OneOffTimerNodedemo_nodes_cpp::ReuseTimerNodedemo_nodes_cpp::ServerNodedemo_nodes_cpp::ClientNodedemo_nodes_cpp::IntrospectionServiceNodedemo_nodes_cpp::IntrospectionClientNodedemo_nodes_cpp::ListParametersdemo_nodes_cpp::ParameterBlackboarddemo_nodes_cpp::SetAndGetParametersdemo_nodes_cpp::ParameterEventsAsyncNodedemo_nodes_cpp::EvenParameterNodedemo_nodes_cpp::SetParametersCallbackdemo_nodes_cpp::ContentFilteringPublisherdemo_nodes_cpp::ContentFilteringSubscriberdemo_nodes_cpp::Talkerdemo_nodes_cpp::LoanedMessageTalkerdemo_nodes_cpp::SerializedMessageTalkerdemo_nodes_cpp::Listenerdemo_nodes_cpp::SerializedMessageListenerdemo_nodes_cpp::ListenerBestEffort
tf2_rostf2_ros::StaticTransformBroadcasterNode
action_tutorials_cppaction_tutorials_cpp::FibonacciActionClientaction_tutorials_cpp::FibonacciActionServer
image_toolsimage_tools::Cam2Imageimage_tools::ShowImage
joyjoy::Joyjoy::GameController
depthimage_to_laserscandepthimage_to_laserscan::DepthImageToLaserScanROS
运行时组合示例(ROS的发布者/订阅者)

        应用程序运行时动态加载/卸载组件,灵活性好。

        打开第一个终端,启动组件容器:

$ros2 run rclcpp_components component_container

        打开第二个终端,检查确认组件容器是否正常运行中:

$ros2 component list

如果返回了/ComponentManager,则说明OK。再在此终端中加载talker组件(源码):

$ros2 component load /ComponentManager composition composition::Talker

返回的结果中包含了该组件的名称以及其对应的ID号:

Loaded component 1 into '/ComponentManager' container node as '/talker'

 

现在我们再切换到第一个终端窗口(加载组件容器),看看里面有些什么内容出来:

可以看到,talker组件加载成功并且在不断发布消息。

        我们再在第二个终端加载listener组件(源码),命令类似于talker,如下:

$ros2 component load /ComponentManager composition composition::Listener

也返回了类似的内容,组件名称listener以及ID号码:

Loaded component 2 into '/ComponentManager' container node as '/listener'

 

我们再检查一下当前的组件列表:

$ros2 component list

运行时组合示例(ROS的服务端和客户端)

        同上面的发布者/订阅者例子,先开启一个终端加载组件容器:

$ros2 run rclcpp_components component_container

在第二个终端依次加载服务端组件(源码)和客户端组件(源码):

$ros2 component load /ComponentManager composition composition::Server
$ros2 component load /ComponentManager composition composition::Client

我们看看第一个终端会有什么内容:

编译时组合示例(硬编码)

        编译ROS 2应用程序时,通过硬编码的方式将节点组合在一起,编译时组合通常意味着在源代码中直接定义和实例化节点对象,并将它们组合成一个或多个进程。这种方式的优点包括简单性和确定性——编译后的应用程序结构是固定的,没有运行时的不确定性。然而,它也降低了灵活性,因为一旦应用程序被编译,节点的组合方式就不能轻易改变。

        这个示例显示,可以复用相同的共享库来编译运行多个组件的单个可执行文件,而无需使用ROS接口。可执行文件包含上面的所有四个组件:发送器和侦听器以及服务器和客户端,它们在主函数中进行了硬编码。

        我们可以通过下面的命令调用:

$ros2 run composition manual_composition

我们可以看看终端返回的内容:

编译组合的方式的组件是没法通过ros2 component list命令查看的。

运行时组合示例(使用dlopen)

        在运行时组合示例中,当我们运行组件容器后并启动服务端/客户端时,可以看到在容器的终端输出中有打印出加载对应的so库并且找到对应的类最终实例化。

        这个利用dlopen的示例通过创建一个通用容器进程并显式地传递库以加载而不使用ROS接口,提供了一种运行时组合的替代方案。该进程将打开每个库,并在库中创建每个“rclcpp::Node”类的一个实例(源码),与前面的流程大同小异,只不过不用那么麻烦开启好几个终端而已。

$ros2 run composition dlopen_composition `ros2 pkg prefix composition`/lib/libtalker_component.so `ros2 pkg prefix composition`/lib/liblistener_component.so

  

可以与上面的截图比较一下,是不是差不多。

dlopen-composed组件通过ros2 component list命令也是发现不了的。

launch方式启动组合

        上面的例子都是逐个命令运行组合,可以比较方便地对过程进行调试、诊断,但是我们也可以一条命令启动多个组件(源码):

$ros2 launch composition composition_demo_launch.py
mike@mike-virtual-machine:~/Desktop/ros2_ws$ ros2 launch composition composition_demo_launch.py
[INFO] [launch]: All log files can be found below /home/mike/.ros/log/2024-04-15-21-18-04-617692-mike-virtual-machine-30402
[INFO] [launch]: Default logging verbosity is set to INFO
[INFO] [component_container-1]: process started with pid [30417]
[component_container-1] [INFO] [1713187085.127445213] [my_container]: Load Library: /opt/ros/iron/lib/libtalker_component.so
[component_container-1] [INFO] [1713187085.128648991] [my_container]: Found class: rclcpp_components::NodeFactoryTemplate<composition::Talker>
[component_container-1] [INFO] [1713187085.128759649] [my_container]: Instantiate class: rclcpp_components::NodeFactoryTemplate<composition::Talker>
[INFO] [launch_ros.actions.load_composable_nodes]: Loaded node '/talker' in container '/my_container'
[component_container-1] [INFO] [1713187085.143832047] [my_container]: Load Library: /opt/ros/iron/lib/liblistener_component.so
[component_container-1] [INFO] [1713187085.144882187] [my_container]: Found class: rclcpp_components::NodeFactoryTemplate<composition::Listener>
[component_container-1] [INFO] [1713187085.144981253] [my_container]: Instantiate class: rclcpp_components::NodeFactoryTemplate<composition::Listener>
[INFO] [launch_ros.actions.load_composable_nodes]: Loaded node '/listener' in container '/my_container'
[component_container-1] [INFO] [1713187086.141280013] [talker]: Publishing: 'Hello World: 1'
[component_container-1] [INFO] [1713187086.141662411] [listener]: I heard: [Hello World: 1]
[component_container-1] [INFO] [1713187087.141013738] [talker]: Publishing: 'Hello World: 2'
[component_container-1] [INFO] [1713187087.141367812] [listener]: I heard: [Hello World: 2]
[component_container-1] [INFO] [1713187088.141100795] [talker]: Publishing: 'Hello World: 3'
[component_container-1] [INFO] [1713187088.141427367] [listener]: I heard: [Hello World: 3]
[component_container-1] [INFO] [1713187089.141335387] [talker]: Publishing: 'Hello World: 4'
[component_container-1] [INFO] [1713187089.141511508] [listener]: I heard: [Hello World: 4]

更多用法

        上面都是关于组合的一些基础用法,下面我们来show几个高级用法。

卸载组件

        再来一遍上面的组合示例,方便用来演示卸载的步骤。

        1.启动组件容器

$ros2 run rclcpp_components component_container

        2.检查确认容器是否正常运行

$ros2 component list

        3.启动talker和listener

$ros2 component load /ComponentManager composition composition::Talker
$ros2 component load /ComponentManager composition composition::Listener

 可以看到每个组件成功加载后都会有一个对应的ID,上面1对应Listener,2对应Talker。

        4.卸载组件

$ros2 component unload /ComponentManager 1 2

 

根据对应的ID来卸载组件,干净明了。

重映射容器及命名空间名字

        我们可以通过下面的命令对组件容器和命名空间的名字进行重映射:

$ros2 run rclcpp_components component_container --ros-args -r __node:=MyContainer -r __ns:=/ns

        第二个终端加载组件:

$ros2 component load /ns/MyContainer composition composition::Listener
重映射组件及命名空间名字

        首先启动组件容器:

$ros2 run rclcpp_components component_container

        我们有几个重映射例子。

        重映射节点名字:

$ros2 component load /ComponentManager composition composition::Talker --node-name talker2

        重映射命名空间:

$ros2 component load /ComponentManager composition composition::Talker --node-namespace /ns

        重映射节点及命名空间:

$ros2 component load /ComponentManager composition composition::Talker --node-name talker3 --node-namespace /ns2

        我们再来检查一下组件列表:

$ros2 component list

将参数值传递到组件

        ros2组件加载命令行支持在构建节点时将任意参数传递给节点。此功能可按如下方式使用:

$ros2 component load /ComponentManager image_tools image_tools::Cam2Image -p burger_mode:=true
将附加参数传递到组件

        ros2组件加载命令行支持将特定选项传递给组件管理器,以便在构建节点时使用。到目前为止,唯一支持的命令行选项是使用进程内通信实例化节点。此功能可按如下方式使用:

$ros2 component load /ComponentManager composition composition::Talker -e use_intra_process_comms:=true

作为共享库的组合节点

        如果要从包中将可组合节点导出为共享库,并在另一个进行链接时组合的包中使用该节点,请将代码添加到CMake文件中,该文件将导入下游包中的实际目标。然后安装生成的文件并导出生成的文件。一个参考例子:ROS Discourse - Ament best practice for sharing libraries

合成非节点衍生组件

        在ROS 2中,组件允许更有效地使用系统资源,并提供了一个强大的功能,使您能够创建不与特定节点绑定的可重用功能。

        使用组件的一个优点是,它们允许我们将非节点派生的功能创建为独立的可执行文件或共享库,这些可执行文件可根据需要加载到ROS系统中。

  1. 要创建不是从节点派生的组件,我们要遵循以下准则:
  2. 构造函数带const rclcpp::NodeOptions&参数;
  3. 实现get_node_base_interface()方法,该方法应返回NodeBaseInterface::SharedPtr。我们可以使用在构造函数中创建的节点的get_node_base_interface()方法来提供此接口。

        可以参考这个不是从节点派生的组件示例,该组件侦听ROS主题:node_like_listener_component。

        内容有点多,慢慢消化。

本篇完。

这篇关于ROS 2边学边练(25)-- 将多个节点组合到一个进程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL多实例管理如何在一台主机上运行多个mysql

《MySQL多实例管理如何在一台主机上运行多个mysql》文章详解了在Linux主机上通过二进制方式安装MySQL多实例的步骤,涵盖端口配置、数据目录准备、初始化与启动流程,以及排错方法,适用于构建读... 目录一、什么是mysql多实例二、二进制方式安装MySQL1.获取二进制代码包2.安装基础依赖3.清

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

C++链表的虚拟头节点实现细节及注意事项

《C++链表的虚拟头节点实现细节及注意事项》虚拟头节点是链表操作中极为实用的设计技巧,它通过在链表真实头部前添加一个特殊节点,有效简化边界条件处理,:本文主要介绍C++链表的虚拟头节点实现细节及注... 目录C++链表虚拟头节点(Dummy Head)一、虚拟头节点的本质与核心作用1. 定义2. 核心价值二

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔