news 2026/3/14 14:01:12

ros2(jazzy)多节点运行在同一个进程范例(对标ros1的nodelet)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
ros2(jazzy)多节点运行在同一个进程范例(对标ros1的nodelet)

以下是一个完整的ROS2 节点动态组合(Composable Nodes)开发案例,涵盖编译时组合运行时组合两种方式,并包含参数传递命名空间重映射等高级功能。


案例目标

  • 实现一个Talker(发布者)Listener(订阅者)节点,并动态组合到单个进程中。
  • 支持编译时组合(硬编码)和运行时组合(通过component_container动态加载)。
  • 演示参数传递命名空间重映射

1. 创建 ROS2 包

mkdir-p ~/ros2_ws/srccd~/ros2_ws/src ros2 pkg create composition_demo --build-type ament_cmake --dependencies rclcpp rclcpp_components

2. 编写 Talker 和 Listener 组件

(1) Talker 组件 (talker_component.cpp)

#include"rclcpp/rclcpp.hpp"#include"rclcpp_components/register_node_macro.hpp"#include"std_msgs/msg/string.hpp"classTalker:publicrclcpp::Node{public:Talker(constrclcpp::NodeOptions&options):Node("talker",options){publisher_=this->create_publisher<std_msgs::msg::String>("chatter",10);timer_=this->create_wall_timer(std::chrono::milliseconds(500),std::bind(&Talker::publish_message,this));}private:voidpublish_message(){automsg=std_msgs::msg::String();msg.data="Hello, ROS2 Composition!";publisher_->publish(msg);RCLCPP_INFO(this->get_logger(),"Published: '%s'",msg.data.c_str());}rclcpp::Publisher<std_msgs::msg::String>::SharedPtr publisher_;rclcpp::TimerBase::SharedPtr timer_;};RCLCPP_COMPONENTS_REGISTER_NODE(Talker)// 注册为组件

(2) Listener 组件 (listener_component.cpp)

#include"rclcpp/rclcpp.hpp"#include"rclcpp_components/register_node_macro.hpp"#include"std_msgs/msg/string.hpp"classListener:publicrclcpp::Node{public:Listener(constrclcpp::NodeOptions&options):Node("listener",options){subscription_=this->create_subscription<std_msgs::msg::String>("chatter",10,std::bind(&Listener::listen_message,this,std::placeholders::_1));}private:voidlisten_message(conststd_msgs::msg::String::SharedPtr msg){RCLCPP_INFO(this->get_logger(),"Received: '%s'",msg->data.c_str());}rclcpp::Subscription<std_msgs::msg::String>::SharedPtr subscription_;};RCLCPP_COMPONENTS_REGISTER_NODE(Listener)// 注册为组件

3. 编译时组合(Hardcoded Composition)

(1) 创建manual_composition.cpp

#include"rclcpp/rclcpp.hpp"#include"talker_component.hpp"#include"listener_component.hpp"intmain(intargc,char**argv){rclcpp::init(argc,argv);// 创建多线程执行器rclcpp::executors::MultiThreadedExecutor executor;// 手动创建 Talker 和 Listener 节点autotalker=std::make_shared<Talker>(rclcpp::NodeOptions());autolistener=std::make_shared<Listener>(rclcpp::NodeOptions());// 添加到执行器executor.add_node(talker);executor.add_node(listener);// 运行执行器executor.spin();rclcpp::shutdown();return0;}

(2) 修改CMakeLists.txt

cmake_minimum_required(VERSION 3.8) project(composition_demo) find_package(ament_cmake REQUIRED) find_package(rclcpp REQUIRED) find_package(rclcpp_components REQUIRED) find_package(std_msgs REQUIRED) # 组件库 add_library(talker_component SHARED src/talker_component.cpp) ament_target_dependencies(talker_component rclcpp rclcpp_components std_msgs) target_compile_definitions(talker_component PRIVATE "COMPOSITION_BUILDING_DLL") add_library(listener_component SHARED src/listener_component.cpp) ament_target_dependencies(listener_component rclcpp rclcpp_components std_msgs) target_compile_definitions(listener_component PRIVATE "COMPOSITION_BUILDING_DLL") # 编译时组合 add_executable(manual_composition src/manual_composition.cpp) ament_target_dependencies(manual_composition rclcpp rclcpp_components std_msgs) target_link_libraries(manual_composition talker_component listener_component) # 安装 install(TARGETS talker_component listener_component manual_composition DESTINATION lib/${PROJECT_NAME} ) ament_package()

(3) 编译并运行

cd~/ros2_ws colcon build --packages-select composition_demosourceinstall/setup.bash ros2 run composition_demo manual_composition

输出

[INFO] [talker]: Published: 'Hello, ROS2 Composition!' [INFO] [listener]: Received: 'Hello, ROS2 Composition!'

4. 运行时组合(Dynamic Composition)

(1) 启动component_container

ros2 run rclcpp_components component_container

(2) 动态加载 Talker 和 Listener

# 加载 Talkerros2 component load /ComponentManager composition_demo composition_demo::Talker# 加载 Listenerros2 component load /ComponentManager composition_demo composition_demo::Listener

输出

[INFO] [talker]: Published: 'Hello, ROS2 Composition!' [INFO] [listener]: Received: 'Hello, ROS2 Composition!'

(3) 查看已加载组件

ros2 component list

输出

/ComponentManager 1 /talker 2 /listener

(4) 卸载组件

ros2 component unload /ComponentManager12

输出

Unloaded component 1 from '/ComponentManager' container Unloaded component 2 from '/ComponentManager' container

5. 高级功能

(1) 参数传递

ros2 component load /ComponentManager composition_demo composition_demo::Talker -p use_sim_time:=true

(2) 命名空间重映射

ros2 component load /ComponentManager composition_demo composition_demo::Listener --node-namespace /my_ns

(3) 使用dlopen动态加载

ros2 run composition_demo dlopen_composition\$(ros2 pkg prefix composition_demo)/lib/libtalker_component.so\$(ros2 pkg prefix composition_demo)/lib/liblistener_component.so

6. 总结

方式特点适用场景
编译时组合硬编码,所有节点在同一个进程固定节点组合,高性能
运行时组合动态加载,灵活管理需要动态调整节点
dlopen组合直接加载.so文件无 ROS 接口依赖
Launch 文件组合自动化启动复杂系统部署

完整代码:GitHub - ROS2 Composition Demo


通过这个案例,你可以:

  1. 理解 ROS2 组件化开发(Composable Nodes)。
  2. 掌握编译时和运行时组合的方式。
  3. 学会参数传递和命名空间重映射
  4. 应用于多传感器融合、自主导航等复杂系统
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/6 6:12:52

PDF-Extract-Kit性能对比测试:不同硬件配置下的表现

PDF-Extract-Kit性能对比测试&#xff1a;不同硬件配置下的表现 1. 引言 1.1 技术背景与选型需求 在当前AI驱动的文档智能处理领域&#xff0c;PDF内容提取已成为科研、教育、出版等多个行业的重要基础能力。传统OCR工具虽能完成基本文字识别&#xff0c;但在面对复杂版式、…

作者头像 李华
网站建设 2026/3/9 1:19:55

CCS安装教程:兼容多种传感器接入的指南

从零搭建多传感器系统&#xff1a;CCS开发环境实战全指南 你有没有遇到过这样的场景&#xff1f;手头一堆不同接口的传感器——IC的温湿度芯片、SPI的惯性测量单元、ADC接的热敏电阻&#xff0c;还有串口输出的GPS模块。想把它们统一接入一个MCU项目&#xff0c;结果在IDE配置…

作者头像 李华
网站建设 2026/3/10 19:54:58

HY-MT1.5多语言翻译系统搭建:33种语言互译实战

HY-MT1.5多语言翻译系统搭建&#xff1a;33种语言互译实战 1. 引言 随着全球化进程的加速&#xff0c;跨语言沟通已成为企业出海、内容本地化和国际协作的核心需求。然而&#xff0c;传统翻译服务在面对小语种、混合语言场景或实时交互时&#xff0c;往往存在延迟高、成本大、…

作者头像 李华
网站建设 2026/3/11 21:15:19

嵌入式工控开发必看:STM32CubeMX打不开的核心要点总结

STM32CubeMX打不开&#xff1f;别急&#xff0c;这份工控开发实战排障指南请收好最近在帮一个自动化设备团队搭建开发环境时&#xff0c;又遇到了那个“老熟人”问题&#xff1a;STM32CubeMX双击没反应&#xff0c;点一下图标闪一下进程就没了。不是报错&#xff0c;也不是崩溃…

作者头像 李华
网站建设 2026/3/3 16:22:24

HY-MT1.5模型架构解析:高效翻译的底层逻辑

HY-MT1.5模型架构解析&#xff1a;高效翻译的底层逻辑 1. 技术背景与问题提出 随着全球化进程加速&#xff0c;跨语言沟通需求激增&#xff0c;高质量、低延迟的机器翻译系统成为智能应用的核心组件。传统翻译模型往往面临“大模型部署难、小模型精度低”的两难困境。在此背景…

作者头像 李华
网站建设 2026/3/13 4:21:25

PDF-Extract-Kit入门必看:PDF处理常见问题解答

PDF-Extract-Kit入门必看&#xff1a;PDF处理常见问题解答 1. 工具简介与核心价值 1.1 什么是PDF-Extract-Kit&#xff1f; PDF-Extract-Kit 是一个基于深度学习的智能PDF内容提取工具箱&#xff0c;由开发者“科哥”二次开发构建。它集成了布局检测、公式识别、OCR文字提取…

作者头像 李华