文章目录
- 一、学习资料:
- 1.1、说明
- 1.2、Autoware Documentation
- 1.3、Autoware Universe Documentation
- 1.4、总结
- 二、概述
- 三、ros2--节点组件
- 什么是组件容器
- 组件的实现原理
- 可组合节点--节点组件
- 什么是节点组件
- 为什么需要可组合节点
- 创建可组合节点时构造函数为什么需要rclcpp::NodeOption的参数
- 可组合节点的实现
- 1,创建可组合节点功能包
- 2,添加节点对象构造参数
- 3,去main,注册可组合节点
- 4,修改可组合节点CMakeLists.txt
- component_container工具
- 作用
- launch启动组件
- luanch文件定义
一、学习资料:
ros2官网文档:http://docs.ros.org/en/humble/
autoware官网文档:https://autowarefoundation.github.io/autoware-documentation/main/
autoware.universe官网文档:https://autowarefoundation.github.io/autoware.universe/main/
1.1、说明
autoware的文档分为2部分,一部分是通用的部分,一部分是详细介绍Universe的实现细节。
Autoware Documentation 主要是一些通用部分,例如安装、使用、设计文档等。
Autoware Universe Documentation 主要是介绍自动驾驶功能模块,例如定位、感知、规划、控制的实现等
1.2、Autoware Documentation
Installation 页面解释了Autoware和相关工具的安装步骤。
Tutorials 页面解释了安装后您应该尝试的几个教程。
How-to guides 页面解释了您在习惯 Autoware 后应该阅读的高级主题。
Design 页面解释了Autoware的设计理念.
Contributing 页面解释了如何为 Autoware 做出贡献.
Datasets 页面包含有关可与 Autoware 一起使用的数据集的信息.
Support 页面解释了一些支持资源.
Competitions 页面解释了与 AWF 相关的持续挑战
1.3、Autoware Universe Documentation
Common 存放一些公共模块
Control 控制模块
Evaluator 评测模块,用于评测自动驾驶一些场景
Launch 启动模块,用于启动自动驾驶的各个模块
Localization 定位模块,用于输出定位信息
Map 地图模块,支持Lanelet2地图格式
Perception 感知模块,用于检测障碍物
Planning 规划模块,用于输出规划轨迹
Sensing 传感器模块,用于接入传感器的消息
Simulator 仿真模块,用于自动驾驶仿真
System 系统模块
Tools 工具模块
Vehicle 车辆模块,提供车辆的转向、动力、制动的控制接口
1.4、总结
总结一下如何看autoware的文档。
autoware是基于ros的,如果想要了解ros和工具,可以参考ros的文档;如果想了解autoware如何安装和使用,可以参考Autoware Documentation;如果想了解autoware自动驾驶各个模块的具体实现,可以参考Autoware Universe Documentation。
二、概述
三、ros2–节点组件
参考ros2官网:https://docs.ros.org/en/galactic/Tutorials/Intermediate/Composition.html#unloading-components
什么是组件容器
在不同的进程中运行多个节点。这样可以使不同的进程独立开。一个崩溃其他可以正常运行。也更方便调试各个节点。
在同一个进程中运行多个节点。这样可以使得通信更加高效。
组件容器功能就是用来实现将多个包中的多个节点放于同一个进程中执行。
组件容器本质上也是一个包中的一个可执行文件而已,这个可执行文件会启动所有加载到这个组件容器中的节点。
ros2 run rclcpp_components component_container
组件的实现原理
单进程运行所有节点。
可组合节点–节点组件
什么是节点组件
将节点对象(插件)使用rclcpp_components库中的宏RCLCPP_COMPONENTS_REGISTER_NODE()进行注册到一个组件中,最后通过cmake构建将这个组件生成共享库;在通过component_container容器工具就可以使用这个组件,component_container容器工具可以自动加载共享库,并启动装在共享库中的组件的节点。
为什么需要可组合节点
为了单进程通信,提高通信效率。
创建可组合节点时构造函数为什么需要rclcpp::NodeOption的参数
节点对象包含了很多可配置的属性,包括节点名称、命名空间、参数等,这些选项可以在运行时根据需要进行动态调整。通过将rclcpp::NodeOption作为构造函数的参数,我们可以在创建节点时,为其指定特定的选项。
在定义可组合节点时,使用rclcpp::NodeOption构造参数也是为了运行创建节点时指定节点的参数。
可组合节点的实现
1,创建可组合节点功能包
ros2 pkg create custom_talker --build-type ament_cmake --dependencies rclcpp rclcpp_components --destination-directory src --node-name talker
2,添加节点对象构造参数
Talker::Talker(const rclcpp::NodeOptions & options): Node("talker", options), count_(0){ pub_ = create_publisher<std_msgs::msg::String>("chatter", 10);
timer_ = create_wall_timer(1s, std::bind(&Talker::on_timer, this));
}
3,去main,注册可组合节点
#include "custom_talker/talker.hpp"
#include <chrono>
#include <iostream>
#include <memory>
#include <utility>
#include "rclcpp/rclcpp.hpp"
#include "std_msgs/msg/string.hpp"
using namespace std::chrono_literals;
namespace custom_talker{
Talker::Talker(const rclcpp::NodeOptions & options): Node("talker", options), count_(0){ pub_ = create_publisher<std_msgs::msg::String>("chatter", 10); timer_ = create_wall_timer(1s, std::bind(&Talker::on_timer, this));
}
void Talker::on_timer(){ auto msg = std::make_unique<std_msgs::msg::String>();
msg->data = "Hello World: " + std::to_string(++count_); RCLCPP_INFO(this->get_logger(), "Publishing: '%s'", msg->data.c_str()); std::flush(std::cout); pub_->publish(std::move(msg));
}
} // namespace custom_talker
#include "rclcpp_components/register_node_macro.hpp"
RCLCPP_COMPONENTS_REGISTER_NODE(custom_talker::Talker)
RCLCPP_COMPONENTS_REGISTER_NODE()
中的参数指定----命名空间::类;
4,修改可组合节点CMakeLists.txt
cmake_minimum_required(VERSION 3.8)
project(custom_talker)
if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
add_compile_options(-Wall -Wextra -Wpedantic)
endif()
#1--添加依赖包
# find dependencies
find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)
find_package(rclcpp_components REQUIRED)
include_directories(include)
#2--将组件生成共享库
#add_executable(talker src/talker.cpp)替换
add_library(talker_component SHARED src/talker.cpp)
target_include_directories(talker_component PUBLIC
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>
$<INSTALL_INTERFACE:include>)
target_compile_features(talker_component PUBLIC c_std_99 cxx_std_17) # Require C99 and C++17
ament_target_dependencies(
talker_component
"rclcpp"
"std_msgs"
"rclcpp_components"
)
#3--必须注册组件
rclcpp_components_register_nodes(talker_component "custom_talker::Talker")
#4--安装组件所在动态库
# install(TARGETS talker
# DESTINATION lib/${PROJECT_NAME})
install(TARGETS
talker_component
ARCHIVE DESTINATION lib
LIBRARY DESTINATION lib
RUNTIME DESTINATION bin)
if(BUILD_TESTING)
find_package(ament_lint_auto REQUIRED)
# the following line skips the linter which checks for copyrights
# comment the line when a copyright and license is added to all source files
set(ament_cmake_copyright_FOUND TRUE)
# the following line skips cpplint (only works in a git repo)
# comment the line when this package is in a git repo and when
# a copyright and license is added to all source files
set(ament_cmake_cpplint_FOUND TRUE)
ament_lint_auto_find_test_dependencies()
endif()
ament_package()
rclcpp_componets_register_nodes()
第一个参数:
是组件的名称,和生成的共享库同名。
第二个参数:
被注册到组件中的插件对象,也就是类对象,如果对象存在于命名空间中,需要指定命名空间。
rclcpp_components_register_nodes(talker_component "custom_talker::Talker")
注意:生成的库一定要加上SHARED生成共享库。
component_container工具
作用
1,启动组件容器工具
ros2 run rclcpp_components component_container
2,查看容器是否成功运行
$ ros2 component list
出现:
/ComponentManager
则成功运行。
3,将包中的插件对象加载到组件容器中
ros2 component load /ComponentManager custom_talker custom_talker::Talker
加载成功会返回成功加载的组件的索引。
参数:
custom_talker-----组件所在包的名称;
custom_talker::Talker—指定的包中的插件的插件名称,也就是类对象名称;
(这个命令并没有指定组件名称,博主猜测应该是通过插件的名称去加载指定的组件到组件容器中。)
组件名称:talker_component
rclcpp_components_register_nodes(talker_component "custom_talker::Talker")
4,使用组件id从容器中卸载组件
ros2 component unload /ComponentManager 1 2
1 2表示卸载两个组件。
launch启动组件
luanch文件定义
1,定义launch文件:
import launch
from launch_ros.actions import ComposableNodeContainer
from launch_ros.descriptions import ComposableNode
def generate_launch_description():
"""Generate launch description with multiple components."""
#--------第一层--定义容器对象--------------#
container = ComposableNodeContainer(
name='my_container',
namespace='',
package='rclcpp_components',
executable='component_container',
#-------------定义节点组件描述对象------------------#
composable_node_descriptions=[
#------定义所有需要在一个进程中使用的节点组件--------#
ComposableNode(
package='composition',
plugin='composition::Talker',
name='talker'),
ComposableNode(
package='composition',
plugin='composition::Listener',
name='listener')
],
output='screen',
)
return launch.LaunchDescription([container])
容器对象的name:
就是启动容器后,容器的名称:
2,安装launch目录
所有的launch功能包在创建launch文件之后,cmake中只需要install launch目录即可:
cmake_minimum_required(VERSION 3.8)
project(component_launch)
if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
add_compile_options(-Wall -Wextra -Wpedantic)
endif()
# find dependencies
find_package(ament_cmake REQUIRED)
# uncomment the following section in order to fill in
# further dependencies manually.
# find_package(<dependency> REQUIRED)
if(BUILD_TESTING)
find_package(ament_lint_auto REQUIRED)
# the following line skips the linter which checks for copyrights
# comment the line when a copyright and license is added to all source files
set(ament_cmake_copyright_FOUND TRUE)
# the following line skips cpplint (only works in a git repo)
# comment the line when this package is in a git repo and when
# a copyright and license is added to all source files
set(ament_cmake_cpplint_FOUND TRUE)
ament_lint_auto_find_test_dependencies()
endif()
#只需要加这个指令即可
install(
DIRECTORY launch
DESTINATION share/${PROJECT_NAME}
)
ament_package()
描述了组件需要启动的所有节点,之后就可以一个指令运行组件,不再需要一步一步启动容器,加载节点组件:
ros2 launch component_launch talker_listener.launch.py