【ROS2】高级:解锁 Fast DDS 中间件的潜力 [社区贡献]

目标:本教程将展示如何在 ROS 2 中使用 Fast DDS 的扩展配置功能

 教程级别:高级

 时间:20 分钟

 目录

  •  背景

  •  先决条件

  • 在同一个节点中混合同步和异步发布

    • 创建具有发布者的节点

    • 创建包含配置文件的 XML 文件

    • 执行发布者节点

    • 创建一个包含订阅者的节点

    • 执行订阅者节点

    • 示例分析

  • 使用其他 FastDDS 功能与 XML

    • 限制匹配订阅者的数量

    • 在主题内使用分区

  • 配置服务和客户端

    • 使用服务和客户端创建节点

    • 为服务和客户端创建 XML 配置文件

    • 执行节点

 背景

ROS 2 堆栈和 Fast DDS 之间的接口由 ROS 2 中间件实现 rmw_fastrtps 提供。此实现可在所有 ROS 2 发行版中使用,无论是从二进制文件还是从源代码。

ROS 2 RMW 仅允许配置某些中间件 QoS(参见 ROS 2 QoS 策略 https://docs.ros.org/en/jazzy/Concepts/Intermediate/About-Quality-of-Service-Settings.html )。然而, rmw_fastrtps 提供了扩展的配置功能,以充分利用 Fast DDS 中的功能。本教程将通过一系列示例指导您如何使用 XML 文件解锁此扩展配置。

为了获得有关在 ROS 2 上使用 Fast DDS 的更多信息,请查看以下文档。https://fast-dds.docs.eprosima.com/en/latest/fastdds/ros2/ros2.html

5b7a953ff3efb2f4844d24ef9b3eea1f.png

sudo apt install ros-jazzy-rmw-fastrtps-cpp
export RMW_IMPLEMENTATION=rmw_fastrtps_cpp
export RMW_IMPLEMENTATION=rmw_fastrtps_dynamic_cpp
RMW_IMPLEMENTATION=rmw_fastrtps_cpp ros2 run <package> <application>
RMW_IMPLEMENTATION=rmw_fastrtps_dynamic_cpp ros2 run <package> <application>

先决条件

本教程假设您知道如何创建一个包。它还假设您知道如何编写一个简单的发布者和订阅者以及一个简单的服务和客户端。尽管示例是用 C++实现的,但相同的概念也适用于 Python 包。

在同一个节点中混合同步和异步发布

在这个第一个例子中,将创建一个具有两个发布者的节点,其中一个是同步发布模式,另一个是异步发布模式。

rmw_fastrtps 默认使用同步发布模式

在同步发布模式下,数据直接在用户线程的上下文中发送。这意味着在写操作期间发生的任何阻塞调用都会阻塞用户线程,从而阻止应用程序继续运行。然而,由于线程之间没有通知或上下文切换,这种模式通常在较低的延迟下产生更高的吞吐量。

另一方面,在异步发布模式下,每次发布者调用写操作时,数据会被复制到队列中,后台线程(异步线程)会收到有关队列中新增数据的通知,并在数据实际发送之前将线程的控制权返回给user。后台线程负责消费队列并将数据发送给每个匹配的reader。

创建带有发布者的节点

首先,在新的工作区上创建一个名为 sync_async_node_example_cpp 的新包:

mkdir -p ~/ros2_ws/src
cd ~/ros2_ws/src
ros2 pkg create --build-type ament_cmake --license Apache-2.0 --dependencies rclcpp std_msgs -- sync_async_node_example_cpp

4332bab791460afb18fd668dfa2ae421.png

然后,向包中添加一个名为 src/sync_async_writer.cpp 的文件,内容如下。请注意,同步发布者将发布在主题 sync_topic 上,而异步发布者将发布在主题 async_topic 上。

#include <chrono> // 包含用于时间操作的头文件
#include <functional> // 包含用于函数对象和绑定的头文件
#include <memory> // 包含用于智能指针的头文件
#include <string> // 包含用于字符串操作的头文件#include "rclcpp/rclcpp.hpp" // 包含ROS 2的C++客户端库
#include "std_msgs/msg/string.hpp" // 包含标准消息类型Stringusing namespace std::chrono_literals; // 使用chrono命名空间中的字面量class SyncAsyncPublisher : public rclcpp::Node // 定义一个名为SyncAsyncPublisher的类,继承自rclcpp::Node
{
public:SyncAsyncPublisher() // 构造函数: Node("sync_async_publisher"), count_(0) // 初始化节点名称为sync_async_publisher,计数器count_初始化为0{// 创建一个同步发布者,发布到主题'sync_topic'sync_publisher_ = this->create_publisher<std_msgs::msg::String>("sync_topic", 10);// 创建一个异步发布者,发布到主题'async_topic'async_publisher_ = this->create_publisher<std_msgs::msg::String>("async_topic", 10);// 定义一个定时器回调函数,每次定时器触发时执行的操作auto timer_callback = this{// 创建一个新的消息auto sync_message = std_msgs::msg::String();sync_message.data = "SYNC: Hello, world! " + std::to_string(count_);// 将消息记录到控制台以显示进度RCLCPP_INFO(this->get_logger(), "Synchronously publishing: '%s'", sync_message.data.c_str());// 使用同步发布者发布消息sync_publisher_->publish(sync_message);// 创建一个新的消息auto async_message = std_msgs::msg::String();async_message.data = "ASYNC: Hello, world! " + std::to_string(count_);// 将消息记录到控制台以显示进度RCLCPP_INFO(this->get_logger(), "Asynchronously publishing: '%s'", async_message.data.c_str());// 使用异步发布者发布消息async_publisher_->publish(async_message);// 准备下一条消息的计数count_++;};// 创建一个定时器,每隔半秒触发一次,执行定时器回调函数timer_ = this->create_wall_timer(500ms, timer_callback);}private:// 定时器,每隔半秒触发一次,发布新的数据rclcpp::TimerBase::SharedPtr timer_;// 异步发布者rclcpp::Publisher<std_msgs::msg::String>::SharedPtr async_publisher_;// 同步发布者rclcpp::Publisher<std_msgs::msg::String>::SharedPtr sync_publisher_;// 已发送的消息数量size_t count_;
};int main(int argc, char * argv[]) // 主函数
{rclcpp::init(argc, argv); // 初始化ROS 2rclcpp::spin(std::make_shared<SyncAsyncPublisher>()); // 创建SyncAsyncPublisher节点并运行rclcpp::shutdown(); // 关闭ROS 2return 0; // 返回0表示程序正常结束
}

现在打开 CMakeLists.txt 文件,添加一个新的可执行文件并将其命名为 SyncAsyncWriter ,以便您可以使用 ros2 run 运行您的节点:

add_executable(SyncAsyncWriter src/sync_async_writer.cpp)
ament_target_dependencies(SyncAsyncWriter rclcpp std_msgs)

最后,添加 install(TARGETS…) 部分,以便 ros2 run 可以找到你的可执行文件:

install(TARGETSSyncAsyncWriterDESTINATION lib/${PROJECT_NAME})

您可以通过删除一些不必要的部分和注释来清理您的 CMakeLists.txt ,使其看起来像这样:

cmake_minimum_required(VERSION 3.8) # 设置CMake的最低版本要求为3.8
project(sync_async_node_example_cpp) # 定义项目名称为sync_async_node_example_cpp# 默认使用C++14标准
if(NOT CMAKE_CXX_STANDARD)set(CMAKE_CXX_STANDARD 14) # 如果没有设置C++标准,则设置为C++14
endif()# 如果使用GNU编译器或Clang编译器,添加编译选项
if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")add_compile_options(-Wall -Wextra -Wpedantic) # 添加编译选项:显示所有警告、额外警告和严格警告
endif()find_package(ament_cmake REQUIRED) # 查找ament_cmake包,标记为必需
find_package(rclcpp REQUIRED) # 查找rclcpp包,标记为必需
find_package(std_msgs REQUIRED) # 查找std_msgs包,标记为必需add_executable(SyncAsyncWriter src/sync_async_writer.cpp) # 添加可执行文件SyncAsyncWriter,源文件为src/sync_async_writer.cpp
ament_target_dependencies(SyncAsyncWriter rclcpp std_msgs) # 设置SyncAsyncWriter的依赖项为rclcpp和std_msgsinstall(TARGETS # 安装目标SyncAsyncWriter # 安装SyncAsyncWriterDESTINATION lib/${PROJECT_NAME}) # 安装路径为lib/${PROJECT_NAME}ament_package() # 声明ament包

如果现在构建并运行此节点,两个发布者将表现相同,两个发布者在主题中都异步发布,因为这是默认的发布模式默认的发布模式配置可以在节点启动期间使用 XML 文件在运行时更改。

创建包含配置文件的 XML 文件

创建一个名为 SyncAsync.xml 的文件,并包含以下内容:

cxy@ubuntu2404-cxy:~/ros2_ws/src/sync_async_node_example_cpp$ gedit SyncAsync.xml
<?xml version="1.0" encoding="UTF-8" ?> <!-- XML声明,定义版本和编码 -->
<profiles xmlns="http://www.eprosima.com/XMLSchemas/fastRTPS_Profiles"> <!-- 定义profiles根元素,并指定其命名空间 --><!-- 默认发布者配置文件 --><publisher profile_name="default_publisher" is_default_profile="true"> <!-- 定义一个发布者配置文件,名称为default_publisher,设置为默认配置文件 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --></publisher><!-- 默认订阅者配置文件 --><subscriber profile_name="default_subscriber" is_default_profile="true"> <!-- 定义一个订阅者配置文件,名称为default_subscriber,设置为默认配置文件 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --></subscriber><!-- sync_topic主题的发布者配置文件 --><publisher profile_name="/sync_topic"> <!-- 定义一个发布者配置文件,名称为/sync_topic --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --><qos> <!-- 定义QoS(服务质量)设置 --><publishMode> <!-- 定义发布模式 --><kind>SYNCHRONOUS</kind> <!-- 设置发布模式为同步 --></publishMode></qos></publisher><!-- async_topic主题的发布者配置文件 --><publisher profile_name="/async_topic"> <!-- 定义一个发布者配置文件,名称为/async_topic --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --><qos> <!-- 定义QoS(服务质量)设置 --><publishMode> <!-- 定义发布模式 --><kind>ASYNCHRONOUS</kind> <!-- 设置发布模式为异步 --></publishMode></qos></publisher></profiles>

请注意,定义了多个发布者和订阅者的配置文件。定义了两个默认配置文件,将 is_default_profile 设置为 true ,以及两个名称与先前定义的主题相符的配置文件: sync_topic 和另一个 async_topic 。这两个配置文件将发布模式分别设置为 SYNCHRONOUS 或 ASYNCHRONOUS 。还请注意,所有配置文件都指定了一个 historyMemoryPolicy 值,这是示例正常运行所需的值,原因将在本教程后面解释。

执行发布者节点

您需要导出以下环境变量以加载 XML:

export RMW_IMPLEMENTATION=rmw_fastrtps_cpp
export RMW_FASTRTPS_USE_QOS_FROM_XML=1
export FASTRTPS_DEFAULT_PROFILES_FILE=~/ros2_ws/src/sync_async_node_example_cpp/SyncAsync.xml

最后,确保您已获取设置文件并运行节点:

source install/setup.bash
ros2 run sync_async_node_example_cpp SyncAsyncWriter

您应该看到发布者从发布节点发送数据,如下所示:

[INFO] [1612972049.994630332] [sync_async_publisher]: Synchronously publishing: 'SYNC: Hello, world! 0'
[INFO] [1612972049.995097767] [sync_async_publisher]: Asynchronously publishing: 'ASYNC: Hello, world! 0'
[INFO] [1612972050.494478706] [sync_async_publisher]: Synchronously publishing: 'SYNC: Hello, world! 1'
[INFO] [1612972050.494664334] [sync_async_publisher]: Asynchronously publishing: 'ASYNC: Hello, world! 1'
[INFO] [1612972050.994368474] [sync_async_publisher]: Synchronously publishing: 'SYNC: Hello, world! 2'
[INFO] [1612972050.994549851] [sync_async_publisher]: Asynchronously publishing: 'ASYNC: Hello, world! 2'

现在你有一个同步发布者和一个异步发布者在同一个节点内运行。

779d7078955599ce9bb7995710049494.png

创建一个带有订阅者的节点

接下来,将创建一个包含订阅者的新节点,这些订阅者将监听 sync_topic 和 async_topic 发布。在名为 src/sync_async_reader.cpp 的新源文件中写入以下内容:

#include <memory> // 包含用于智能指针的头文件#include "rclcpp/rclcpp.hpp" // 包含ROS 2的C++客户端库
#include "std_msgs/msg/string.hpp" // 包含标准消息类型Stringclass SyncAsyncSubscriber : public rclcpp::Node // 定义一个名为SyncAsyncSubscriber的类,继承自rclcpp::Node
{
public:SyncAsyncSubscriber() // 构造函数: Node("sync_async_subscriber") // 初始化节点名称为sync_async_subscriber{// Lambda函数,每次接收到新消息时运行auto topic_callback = this{RCLCPP_INFO(this->get_logger(), "I heard: '%s'", msg.data.c_str()); // 将接收到的消息记录到控制台};// 创建一个同步订阅者,订阅主题'sync_topic'// 并将其绑定到topic_callbacksync_subscription_ = this->create_subscription<std_msgs::msg::String>("sync_topic", 10, topic_callback);// 创建一个异步订阅者,订阅主题'async_topic'// 并将其绑定到topic_callbackasync_subscription_ = this->create_subscription<std_msgs::msg::String>("async_topic", 10, topic_callback);}private:// 一个订阅'sync_topic'主题的订阅者rclcpp::Subscription<std_msgs::msg::String>::SharedPtr sync_subscription_;// 一个订阅'async_topic'主题的订阅者rclcpp::Subscription<std_msgs::msg::String>::SharedPtr async_subscription_;
};int main(int argc, char * argv[]) // 主函数
{rclcpp::init(argc, argv); // 初始化ROS 2rclcpp::spin(std::make_shared<SyncAsyncSubscriber>()); // 创建SyncAsyncSubscriber节点并运行rclcpp::shutdown(); // 关闭ROS 2return 0; // 返回0表示程序正常结束
}

打开 CMakeLists.txt 文件,在前一个 SyncAsyncWriter 下添加一个新的可执行文件,并将其命名为 SyncAsyncReader

add_executable(SyncAsyncReader src/sync_async_reader.cpp)
ament_target_dependencies(SyncAsyncReader rclcpp std_msgs)install(TARGETSSyncAsyncReaderDESTINATION lib/${PROJECT_NAME})

执行订阅者节点

在一个终端中运行发布者节点后,打开另一个终端并导出加载 XML 所需的环境变量:

export RMW_IMPLEMENTATION=rmw_fastrtps_cpp
export RMW_FASTRTPS_USE_QOS_FROM_XML=1
export FASTRTPS_DEFAULT_PROFILES_FILE=~/ros2_ws/src/sync_async_node_example_cpp/SyncAsync.xml

最后,确保您已获取设置文件并运行节点:

source install/setup.bash
ros2 run sync_async_node_example_cpp SyncAsyncReader

您应该看到订阅者从发布节点接收数据,如下所示:

[INFO] [1612972054.495429090] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 10'
[INFO] [1612972054.995410057] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 10'
[INFO] [1612972055.495453494] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 11'
[INFO] [1612972055.995396561] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 11'
[INFO] [1612972056.495534818] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 12'
[INFO] [1612972056.995473953] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 12'

360ce14b2d48f0ad53fef08614cbfae6.png

示例分析

配置文件 XML

XML 文件定义了发布者和订阅者的几种配置。您可以拥有一个默认的发布者配置文件和几个特定主题的发布者配置文件。唯一的要求是所有发布者配置文件必须有不同的名称,并且只能有一个默认配置文件。订阅者也是如此

为了定义特定主题的配置,只需将配置文件命名为 ROS 2 主题名称(如示例中的 /sync_topic 和 /async_topic ), rmw_fastrtps 将此配置文件应用于该主题的所有发布者和订阅者。默认配置文件由属性 is_default_profile 设置为 true 标识,并在没有其他名称与主题名称匹配的配置文件时充当回退配置文件。

环境变量 FASTRTPS_DEFAULT_PROFILES_FILE 用于通知 Fast DDS 配置文件的 XML 文件路径。

RMW_FASTRTPS_USE_QOS_FROM_XML

在所有可配置属性中, rmw_fastrtps 对 publishMode 和 historyMemoryPolicy 的处理方式不同。默认情况下,这些值在 rmw_fastrtps 实现中设置为 ASYNCHRONOUS 和 PREALLOCATED_WITH_REALLOC ,并且 XML 文件中设置的值将被忽略。为了使用 XML 文件中的值,必须将环境变量 RMW_FASTRTPS_USE_QOS_FROM_XML 设置为 1 。

然而,这还涉及另一个警告:如果设置了 RMW_FASTRTPS_USE_QOS_FROM_XML ,但 XML 文件没有定义 publishMode 或 historyMemoryPolicy ,这些属性将采用 Fast DDS 默认值而不是 rmw_fastrtps 默认值。这一点很重要,尤其是对于 historyMemoryPolicy ,因为 Fast DDS 默认值是 PREALLOCATED ,它不适用于 ROS2 主题数据类型。因此,在示例中,已明确设置了该策略的有效值( DYNAMIC )。

rmw_qos_profile_t 的优先级 

ROS 2 QoS 包含在 rmw_qos_profile_t https://docs.ros.org/en/jazzy/p/rmw/generated/structrmw__qos__profile__s.html中的 QoS 始终被遵守,除非设置为 *_SYSTEM_DEFAULT 。在这种情况下,将应用 XML 值(或在没有 XML 值的情况下应用 Fast DDS 默认值)。这意味着,如果 rmw_qos_profile_t 中的任何 QoS 设置为 *_SYSTEM_DEFAULT 以外的值,则 XML 中的相应值将被忽略

d163218813629388980470c8183c57a2.png

使用其他 FastDDS 功能与 XML

虽然我们创建了一个具有不同配置的两个发布者的节点,但很难检查它们的行为是否不同。现在已经介绍了 XML 配置文件的基础知识,让我们使用它们来配置一些对节点有视觉效果的东西。具体来说,将在一个发布者上设置最大匹配订阅者数量,在另一个发布者上设置分区定义。请注意,这些只是通过 XML 文件可以调整的所有配置属性中的一些非常简单的示例。请参阅*Fast DDS*文档https://fast-dds.docs.eprosima.com/en/latest/fastdds/xml_configuration/xml_configuration.html#xml-profiles 以查看可以通过 XML 文件配置的属性的完整列表。

b53a3af64a33798028be51944a2e028e.png

限制匹配订阅者的数量

将最大数量的匹配订阅者添加到 /async_topic 发布者配置文件。它应该看起来像这样:

<!-- async_topic主题的发布者配置文件 -->
<publisher profile_name="/async_topic"> <!-- 定义名为/async_topic的发布者配置文件 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --><qos> <!-- 定义QoS(服务质量)设置 --><publishMode> <!-- 定义发布模式 --><kind>ASYNCHRONOUS</kind> <!-- 设置发布模式为异步 --></publishMode></qos><matchedSubscribersAllocation> <!-- 定义匹配订阅者的分配策略 --><initial>0</initial> <!-- 初始分配的订阅者数量为0 --><maximum>1</maximum> <!-- 最大分配的订阅者数量为1 --><increment>1</increment> <!-- 每次增加的订阅者数量为1 --></matchedSubscribersAllocation>
</publisher>

匹配订阅者的数量被限制为一个。

现在打开三个终端,不要忘记源化设置文件并设置所需的环境变量。在第一个终端上运行发布者节点,在另外两个终端上运行订阅者节点。您应该看到只有第一个订阅者节点接收到来自两个主题的消息。第二个订阅者节点无法在 /async_topic 中完成匹配过程,因为发布者阻止了它,因为它已经达到了匹配发布者的最大数量。因此,只有来自 /sync_topic 的消息将会在这个第三终端中接收到。

[INFO] [1613127657.088860890] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 18'
[INFO] [1613127657.588896594] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 19'
[INFO] [1613127658.088849401] [sync_async_subscriber]: I heard: 'SYNC: Hello, world! 20'

在主题内使用分区

分区功能可用于控制在同一主题内哪些发布者和订阅者交换信息

分区在由域 ID 引起的物理隔离内引入了逻辑实体隔离级别的概念。为了使发布者与订阅者进行通信,他们必须至少属于一个共同的分区。分区代表了在域和主题之外分离发布者和订阅者的另一个级别。与域和主题不同,一个端点可以同时属于多个分区。为了在不同的域或主题上共享某些数据,每个域或主题必须有一个不同的发布者,分享其自己的更改历史。然而,单个发布者可以使用单个主题数据更改在不同的分区上共享相同的数据样本,从而减少网络过载

让我们将 /sync_topic 发布者更改为分区 part1 ,并创建一个使用分区 part2 的新 /sync_topic 订阅者。他们的配置文件现在应如下所示:

<!-- sync_topic主题的发布者配置文件 -->
<publisher profile_name="/sync_topic"> <!-- 定义一个发布者配置文件,名称为/sync_topic --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --><qos> <!-- 定义QoS(服务质量)设置 --><publishMode> <!-- 定义发布模式 --><kind>SYNCHRONOUS</kind> <!-- 设置发布模式为同步 --></publishMode><partition> <!-- 定义分区 --><names> <!-- 分区名称 --><name>part1</name> <!-- 设置分区名称为part1 --></names></partition></qos>
</publisher><!-- sync_topic主题的订阅者配置文件 -->
<subscriber profile_name="/sync_topic"> <!-- 定义一个订阅者配置文件,名称为/sync_topic --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy> <!-- 设置历史内存策略为动态 --><qos> <!-- 定义QoS(服务质量)设置 --><partition> <!-- 定义分区 --><names> <!-- 分区名称 --><name>part2</name> <!-- 设置分区名称为part2 --></names></partition></qos>
</subscriber>

打开两个终端。不要忘记加载设置文件并设置所需的环境变量。在第一个终端上运行发布者节点,在另一个终端上运行订阅者节点。您应该看到只有 /async_topic 消息到达订阅者。 /sync_topic 订阅者没有接收到数据,因为它与相应的发布者在不同的分区中。

[INFO] [1612972054.995410057] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 10'
[INFO] [1612972055.995396561] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 11'
[INFO] [1612972056.995473953] [sync_async_subscriber]: I heard: 'ASYNC: Hello, world! 12'

配置服务和客户端

服务和客户端各有一个发布者和一个订阅者,它们通过两个不同的主题进行通信。例如,对于名为 ping 的服务,有:

  • 在 /rq/ping 上监听请求的服务订阅者。

  • 服务发布者在 /rr/ping 上发送响应。

  • 客户端发布者在 /rq/ping 上发送请求。

  • 一个客户端订阅者正在监听 /rr/ping 上的响应。

尽管您可以使用这些主题名称在 XML 上设置配置文件,有时您可能希望将相同的配置文件应用于节点上的所有服务或客户端。与其为所有服务生成的所有主题名称复制相同的配置文件,您可以只创建一个名为 service 的发布者和订阅者配置文件对。对于创建名为 client 的对的客户端,也可以这样做。

使用服务和客户端创建节点

开始使用该服务创建节点。在您的包中添加一个名为 src/ping_service.cpp 的新源文件,并包含以下内容:

#include <memory> // 包含用于智能指针的头文件#include "rclcpp/rclcpp.hpp" // 包含ROS 2的C++客户端库
#include "example_interfaces/srv/trigger.hpp" // 包含example_interfaces包中的Trigger服务/*** 服务操作:响应success=true并在控制台打印请求*/
void ping(const std::shared_ptr<example_interfaces::srv::Trigger::Request> request,std::shared_ptr<example_interfaces::srv::Trigger::Response> response)
{// 请求数据未使用(void) request;// 构建响应response->success = true;// 记录到控制台RCLCPP_INFO(rclcpp::get_logger("ping_server"), "Incoming request"); // 打印收到请求的日志RCLCPP_INFO(rclcpp::get_logger("ping_server"), "Sending back response"); // 打印发送响应的日志
}int main(int argc, char **argv) // 主函数
{rclcpp::init(argc, argv); // 初始化ROS 2// 创建节点和服务std::shared_ptr<rclcpp::Node> node = rclcpp::Node::make_shared("ping_server"); // 创建名为ping_server的节点rclcpp::Service<example_interfaces::srv::Trigger>::SharedPtr service =node->create_service<example_interfaces::srv::Trigger>("ping", &ping); // 创建名为ping的服务,并绑定到ping函数// 记录服务已准备好的日志RCLCPP_INFO(rclcpp::get_logger("ping_server"), "Ready to serve."); // 打印服务已准备好的日志// 运行节点rclcpp::spin(node); // 运行节点rclcpp::shutdown(); // 关闭ROS 2
}

在名为 src/ping_client.cpp 的文件中创建客户端,内容如下:

#include <chrono> // 包含用于时间操作的头文件
#include <memory> // 包含用于智能指针的头文件#include "rclcpp/rclcpp.hpp" // 包含ROS 2的C++客户端库
#include "example_interfaces/srv/trigger.hpp" // 包含example_interfaces包中的Trigger服务using namespace std::chrono_literals; // 使用chrono命名空间中的字面量int main(int argc, char **argv) // 主函数
{rclcpp::init(argc, argv); // 初始化ROS 2// 创建节点和客户端std::shared_ptr<rclcpp::Node> node = rclcpp::Node::make_shared("ping_client"); // 创建名为ping_client的节点rclcpp::Client<example_interfaces::srv::Trigger>::SharedPtr client =node->create_client<example_interfaces::srv::Trigger>("ping"); // 创建名为ping的客户端// 创建请求auto request = std::make_shared<example_interfaces::srv::Trigger::Request>(); // 创建Trigger服务的请求// 等待服务可用while (!client->wait_for_service(1s)) { // 每隔1秒检查一次服务是否可用if (!rclcpp::ok()) { // 如果ROS 2被中断RCLCPP_ERROR(rclcpp::get_logger("ping_client"), "Interrupted while waiting for the service. Exiting."); // 打印错误日志return 0; // 返回0表示程序正常结束}RCLCPP_INFO(rclcpp::get_logger("ping_client"), "Service not available, waiting again..."); // 打印服务不可用的日志}// 现在服务可用了,发送请求RCLCPP_INFO(rclcpp::get_logger("ping_client"), "Sending request"); // 打印发送请求的日志auto result = client->async_send_request(request); // 异步发送请求// 等待结果并将其记录到控制台if (rclcpp::spin_until_future_complete(node, result) ==rclcpp::FutureReturnCode::SUCCESS) // 如果成功接收到响应{RCLCPP_INFO(rclcpp::get_logger("ping_client"), "Response received"); // 打印接收到响应的日志} else {RCLCPP_ERROR(rclcpp::get_logger("ping_client"), "Failed to call service ping"); // 打印调用服务失败的日志}rclcpp::shutdown(); // 关闭ROS 2return 0; // 返回0表示程序正常结束
}

打开 CMakeLists.txt 文件并添加两个新的可执行文件 ping_service 和 ping_client :

find_package(example_interfaces REQUIRED) # 查找example_interfaces包,标记为必需add_executable(ping_service src/ping_service.cpp) # 添加可执行文件ping_service,源文件为src/ping_service.cpp
ament_target_dependencies(ping_service example_interfaces rclcpp) # 设置ping_service的依赖项为example_interfaces和rclcppadd_executable(ping_client src/ping_client.cpp) # 添加可执行文件ping_client,源文件为src/ping_client.cpp
ament_target_dependencies(ping_client example_interfaces rclcpp) # 设置ping_client的依赖项为example_interfaces和rclcppinstall(TARGETS # 安装目标ping_service # 安装ping_serviceDESTINATION lib/${PROJECT_NAME}) # 安装路径为lib/${PROJECT_NAME}install(TARGETS # 安装目标ping_client # 安装ping_clientDESTINATION lib/${PROJECT_NAME}) # 安装路径为lib/${PROJECT_NAME}

最后,构建包。

为服务和客户端创建 XML 配置文件

创建一个名为 ping.xml 的文件,并包含以下内容:

<?xml version="1.0" encoding="UTF-8" ?>
<profiles xmlns="http://www.eprosima.com/XMLSchemas/fastRTPS_Profiles"><!-- 默认发布者配置文件 --><publisher profile_name="default_publisher" is_default_profile="true"><!-- 历史内存策略设置为动态 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy></publisher><!-- 默认订阅者配置文件 --><subscriber profile_name="default_subscriber" is_default_profile="true"><!-- 历史内存策略设置为动态 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy></subscriber><!-- 服务发布者配置为同步模式 --><publisher profile_name="service"><!-- 历史内存策略设置为动态 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy><qos><publishMode><!-- 发布模式设置为同步 --><kind>SYNCHRONOUS</kind></publishMode></qos></publisher><!-- 客户端发布者配置为异步模式 --><publisher profile_name="client"><!-- 历史内存策略设置为动态 --><historyMemoryPolicy>DYNAMIC</historyMemoryPolicy><qos><publishMode><!-- 发布模式设置为异步 --><kind>ASYNCHRONOUS</kind></publishMode></qos></publisher></profiles>

此配置文件将服务上的发布模式设置为 SYNCHRONOUS ,将客户端上的发布模式设置为 ASYNCHRONOUS 。请注意,我们仅定义了服务和客户端的发布者配置文件,但也可以提供订阅者配置文件。

执行节点

打开两个终端,并在每个终端上加载设置文件。然后设置加载 XML 所需的环境变量:

export RMW_IMPLEMENTATION=rmw_fastrtps_cpp
export RMW_FASTRTPS_USE_QOS_FROM_XML=1
export FASTRTPS_DEFAULT_PROFILES_FILE=~/ros2_ws/src/sync_async_node_example_cpp/ping.xml

在第一个终端上运行服务节点。

ros2 run sync_async_node_example_cpp ping_service

您应该看到服务正在等待请求:

[INFO] [1612977403.805799037] [ping_server]: Ready to serve.

在第二个终端上运行客户端节点。

ros2 run sync_async_node_example_cpp ping_client

您应该看到客户端发送请求并接收响应:

[INFO] [1612977404.805799037] [ping_client]: Sending request
[INFO] [1612977404.825473835] [ping_client]: Response received

同时,服务器控制台中的输出已更新:

[INFO] [1612977403.805799037] [ping_server]: Ready to serve.
[INFO] [1612977404.807314904] [ping_server]: Incoming request
[INFO] [1612977404.836405125] [ping_server]: Sending back response

0913e1ab977a79cb45f4e722dd25d82c.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/47583.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

单例模式->饿汉模式->懒汉模式->阻塞队列->模拟实现阻塞队列->生产者消费者模型

单例模式->是一种固定套路,类似于"棋谱",按照套路来,可以避免一些问题 单例模式的特点->能够保证在某个类中只存在一个实例,不会创建多个实例 饿汉模式(线程安全):最基础的单例模式,类加载的同时就会创建实例,是线程安全的 public class Singleton {// 在类加…

Flutter应用开发:掌握StatefulWidget的实用技巧

前言 随着移动应用的日益复杂&#xff0c;状态管理成为了 Flutter 应用开发中的一项重要挑战。 状态&#xff0c;即应用中的可变数据&#xff0c;它驱动着用户界面的渲染和交互。 在 Flutter 这样的声明式 UI 框架中&#xff0c;如何高效、可维护地管理状态&#xff0c;对于…

GuLi商城-商品服务-API-属性分组-分组修改级联选择器回显

前端代码:略 后端回显接口: 递归方法: @Override publi

jquery中pdf在页面的显示和导出

jquery中pdf在页面的显示和导出 01 显示pdf01 .pdf结尾在线接口显示到页面 &#xff08;pdf.js库怎么安装及使用&#xff09;&#xff1a;只显示一页02 如何用PDF.JS显示整个PDF (而不仅仅是一页)&#xff1f;03 jQuery实现在线预览PDF文件(通过a标签链接跳转)&#xff1a; 02 …

‍我想我大抵是疯了,我喜欢上了写单元测试

前言 大家好我是聪。相信有不少的小伙伴喜欢写代码&#xff0c;但是对于单元测试这些反而觉得多此一举&#xff0c;想着我都在接口文档测过了&#xff01;还要写什么单元测试&#xff01;写不了一点&#xff01;&#xff01; 由于本人也是一个小小程序猿&#x1f649;&#xf…

华为云SQLServer 慢日志查看

作者&#xff1a;梦莱 1、背景 华为云目前只支持 SQLServer 登录数据库&#xff0c;不支持查看慢日志。对于开启慢日志的实例&#xff0c;也只能通过将慢日志下载到本地 再远程连接目标实例数据库查看。本篇将华为云 SQLServer 实例出现资源异常&#xff0c;排查问题的方案整…

【java】力扣 合法分割的最小下标

文章目录 题目链接题目描述思路代码 题目链接 2780.合法分割的最小下标 题目描述 思路 这道题是摩尔算法的一种扩展 我们先可以找到候选人出来&#xff0c;然后去计算他在左右两边元素出现的次数&#xff0c;只有当他左边时&#xff0c;左边出现的次数2 >左边的长度&…

【 LCD1602显示屏】使用STC89C51控制1602显示、读写操作时序

文章目录 LCD1602显示概述&#xff1a;引脚说明控制指令接线 控制思路步骤 代码示例总结对databuffer dataShow;的理解 LCD1602显示 概述&#xff1a; LCD1602&#xff08;Liquid Crystal Display&#xff09;是一种工业字符型液晶&#xff0c;能够同时显示 1602 即 32 字符…

SpringBoot增加网关服务

一、新建gateway项目 二、添加依赖 dependencies {implementation org.springframework.cloud:spring-cloud-starter-gateway:4.0.0 } 三、增加路由规则配置 一个web服务、一个service服务 bootstrap.yaml&#xff1a; server:port: 80 spring:application:name: gatewayc…

【STM32 HAL库】I2S的使用

使用CubeIDE实现I2S发数据 1、配置I2S 我们的有效数据是32位的&#xff0c;使用飞利浦格式。 2、配置DMA **这里需要注意&#xff1a;**i2s的DR寄存器是16位的&#xff0c;如果需要发送32位的数据&#xff0c;是需要写两次DR寄存器的&#xff0c;所以DMA的外设数据宽度设置16…

探索Python自然语言处理的新篇章:jionlp库介绍

探索Python自然语言处理的新篇章&#xff1a;jionlp库介绍 1. 背景&#xff1a;为什么选择jionlp&#xff1f; 在Python的生态中&#xff0c;自然语言处理&#xff08;NLP&#xff09;是一个活跃且不断发展的领域。jionlp是一个专注于中文自然语言处理的库&#xff0c;它提供了…

Ubuntu 安装 XRDP,替代系统自带RDP远程桌面

起因&#xff0c;Ubuntu的自带RDP远程桌面很好用&#xff0c;但很傻卵&#xff0c;必须登录。 而设置了自动登录也不能解开KEYRING&#xff0c;必须必须必须用GUI手动登录。 &#xff08;我远程我用头给你坐机子面前开显示器先登录&#xff1f;&#xff1f;&#xff09; 比起VN…

【HarmonyOS】HarmonyOS NEXT学习日记:三、初识ArkUI

【HarmonyOS】HarmonyOS NEXT学习日记&#xff1a;三、初识ArkUI 忘掉HTML和CSS&#xff0c;ArkUI里构建页面的最小单位就是 “组件”&#xff0c;所以今天的目标就是认识一些常用的基础组件&#xff0c;以及他们的用法&#xff0c;对ArkUI形成一个基本认识。 基本组成 了解…

重塑七星拼团模式:共创互赢新生态

在当今商业模式的洪流中&#xff0c;七星拼团模式凭借其创新的激励机制与深植的互助文化&#xff0c;独树一帜&#xff0c;成为了推动市场活跃与消费者参与的新引擎。本文将重新构思并阐述该模式的三大支柱——直推奖赏、滑落回馈与循环成就奖&#xff0c;同时深入探讨其互助逻…

(error) MOVED 12706 192.168.187.139:6379

Redis操作set、get等操作出现如下错误 (error) MOVED 12706 192.168.187.139:6379 这种情况一般是因为启动 redis-cli 时没有设置集群模式所导致&#xff1b; 在开启集群后&#xff0c;redis-cli用普通用户登录无法操作集群中的数据&#xff0c;需要加上-c 用集群模式登录才可…

网络故障处理及分析工具:Wireshark和Tcpdump集成

Wireshark 是一款免费的开源数据包嗅探器和网络协议分析器&#xff0c;已成为网络故障排除、分析和安全&#xff08;双向&#xff09;中不可或缺的工具。 本文深入探讨了充分利用 Wireshark 的功能、用途和实用技巧。 无论您是开发人员、安全专家&#xff0c;还是只是对网络操…

k8s集群 安装配置 Prometheus+grafana

k8s集群 安装配置 Prometheusgrafana k8s环境如下&#xff1a;机器规划&#xff1a; node-exporter组件安装和配置安装node-exporter通过node-exporter采集数据显示192.168.40.180主机cpu的使用情况显示192.168.40.180主机负载使用情况 Prometheus server安装和配置创建sa账号&…

飞凌全志T527开发板modbus移植使用教程

交叉编译 进入到源码目录&#xff0c;执行 ./configure ac_cv_func_malloc_0_nonnullyes --hostaarch64-none-linux-gnu --enable-static --prefix/home/feng/文档/development/Linux/application/OK527N/libmodbus-3.1.10/install/其中–host为交叉编译器的前缀&#xff1b;…

巧用通义灵码助力护网面试

前言 前几年护网还算是一个比较敏感的话题&#xff0c;但是随着近段时间的常态化开始&#xff0c;护网行动也是逐渐走进了大众的视野&#xff0c;成为了社会各界共同关注的安全盛事。本篇也是受通义灵码备战求职季活动的启发&#xff0c;结合近期要开始的护网行动&#xff0c…

前端面试题(JS篇五)

一、同步与异步的区别 同步指的是当一个进程在执行某一个请求的时候&#xff0c;如果这个请求需要等待一段时间才能返回&#xff0c;那么这个进程会一直等待下去&#xff0c;直到这个消息返回之后才会继续执行。 指的是当一个进程在执行某一个请求的时候&#xff0c;如果这个请…