ROS 2边学边练(16)-- 自定义msg和srv文件

2024-04-07 23:36
文章标签 16 自定义 ros msg 边学边 srv

本文主要是介绍ROS 2边学边练(16)-- 自定义msg和srv文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

        在前面的文章我们在学习主题(topic)和服务(service)通信方法时,使用的一直是ROS 2提供好的消息结构文件(xxx.msg)和服务结构文件(xxx.srv),稀里糊涂的就这样过去了,如果我们有个需求,ROS 2提供的msg和srv无法满足,那我们就得定义自己结构的msg和srv文件了。

动动手

创建一个功能包

        .msg和.srv文件需要分别处于两个文件夹下(msg和srv文件夹,具体功能包目录下同一等级),我们先来创建功能包,进入工作空间根路径(先source install/setup.bash),执行下面命令创建tutorial_interfaces功能包:

$ros2 pkg create --build-type ament_cmake --license Apache-2.0 tutorial_interfaces

 

完成后再进入功能包tutorial_interfaces的根路径下,新建俩个文件夹用来放置.msg和.srv文件:

$mkdir msg srv

这俩文件夹可以被C++实现的节点模块使用,也能被python实现的节点模块使用。

自定义

msg

        首先在msg文件夹下新建一个叫Num.msg的文件,添加下面这行内容到里面:

int64 num

该消息名称为Num,只有一个类型为int64的元素num。

        然后同样在msg文件夹下新建一个叫Sphere.msg的文件,添加下面两行内容到里面:

geometry_msgs/Point center
float64 radius

Sphere消息包含了两个元素,第一个为系统提供的功能包geometry_msgs中Point类型的center(可以看出我们可以嵌入消息),第二个为float64类型的radius。

srv

        在功能包的srv文件夹下(ros2_ws/src/tutorial_interfaces/srv)新建一个叫AddThreeInts.srv的文件,添加下面内容到里面:

int64 a
int64 b
int64 c
---
int64 sum

请求元素有3个:a、b、c,都是int64类型,回复依然是int64类型的sum。

修改CMakeLists.txt

        要将我们自定义的接口转换为特定语言(C++/PYTHON)的代码,我们需要将下面的内容增加到CMakeLists.txt中去:

find_package(geometry_msgs REQUIRED)
find_package(rosidl_default_generators REQUIRED)rosidl_generate_interfaces(${PROJECT_NAME}"msg/Num.msg""msg/Sphere.msg""srv/AddThreeInts.srv"DEPENDENCIES geometry_msgs # Add packages that above messages depend on, in this case geometry_msgs for Sphere.msg
)

对于上面的内容,“rosidl_generate_interfaces()”第一个参数必须是{PROJECT_NAME},否则可能会出现非同一般的错误。

修改package.xml

        将下面内容添加到package.xml内:

<depend>geometry_msgs</depend>
<buildtool_depend>rosidl_default_generators</buildtool_depend>
<exec_depend>rosidl_default_runtime</exec_depend>
<member_of_group>rosidl_interface_packages</member_of_group>

简单解释一下,geometry_msgs在我们自定义的Sphere.msg中有引用,所以需要依赖(库依赖<depend>),rosidl_default_generators会生成语言的代码(C++/PYTHON),其是一种构建工具(工具依赖<buildtool_depend>),所以需要依赖,rosidl_default_runtime是一个运行时或执行阶段需要的依赖项,需要接口能够在以后被使用(执行依赖<exec_depend>),rosidl_interface_packages是功能包tutorial_interfaces应该与之关联的依赖组的名称,使用<member_of_group>标记声明。

 

构建包

        进入工作空间根路径,按照下面的命令进行包的构建工作:

$colcon build --packages-select tutorial_interfaces

确认自定义的msg和srv被成功构建 

        大家应该还记得如何查看msg和srv具体数据结构的命令吧,新开一个终端,先source环境变量如何进行下面操作:

$ros2 interface show tutorial_interfaces/msg/Num

再来看看另外一个消息Sphere和剩下的AddThreeInts服务:

$ros2 interface show tutorial_interfaces/msg/Sphere
$ros2 interface show tutorial_interfaces/srv/AddThreeInts

 

 Perfect!

测试新的接口

Num.msg

        为方便测试,我们利用之前用过的例子(发布者/订阅者)来测试这个新的消息Num.msg,我们在工作空间的src路径下,进入cpp_pubsub文件夹,先修改发布者节点源文件publisher_member_function.cpp,按照下面的内容进行修改:

#include <chrono>
#include <memory>#include "rclcpp/rclcpp.hpp"
#include "tutorial_interfaces/msg/num.hpp"                                            // CHANGEusing namespace std::chrono_literals;class MinimalPublisher : public rclcpp::Node
{
public:MinimalPublisher(): Node("minimal_publisher"), count_(0){publisher_ = this->create_publisher<tutorial_interfaces::msg::Num>("topic", 10);  // CHANGEtimer_ = this->create_wall_timer(500ms, std::bind(&MinimalPublisher::timer_callback, this));}private:void timer_callback(){auto message = tutorial_interfaces::msg::Num();                                   // CHANGEmessage.num = this->count_++;                                                     // CHANGERCLCPP_INFO_STREAM(this->get_logger(), "Publishing: '" << message.num << "'");    // CHANGEpublisher_->publish(message);}rclcpp::TimerBase::SharedPtr timer_;rclcpp::Publisher<tutorial_interfaces::msg::Num>::SharedPtr publisher_;             // CHANGEsize_t count_;
};int main(int argc, char * argv[])
{rclcpp::init(argc, argv);rclcpp::spin(std::make_shared<MinimalPublisher>());rclcpp::shutdown();return 0;
}

原先调用标准消息的语句是:

publisher_ = this->create_publisher<std_msgs::msg::String>("topic", 10);

现在使用自定义消息的语句是:

publisher_ = this->create_publisher<tutorial_interfaces::msg::Num>("topic", 10); 

可以发现<..>里面的内容变化了,另外在install路径下对应的tutorial_interfaces包里面的include路径下有自动生成一个Num.hpp头文件。

再来subscriber_member_function.cpp.c文件,内容如下:

#include <functional>
#include <memory>#include "rclcpp/rclcpp.hpp"
#include "tutorial_interfaces/msg/num.hpp"                                       // CHANGEusing std::placeholders::_1;class MinimalSubscriber : public rclcpp::Node
{
public:MinimalSubscriber(): Node("minimal_subscriber"){subscription_ = this->create_subscription<tutorial_interfaces::msg::Num>(    // CHANGE"topic", 10, std::bind(&MinimalSubscriber::topic_callback, this, _1));}private:void topic_callback(const tutorial_interfaces::msg::Num & msg) const  // CHANGE{RCLCPP_INFO_STREAM(this->get_logger(), "I heard: '" << msg.num << "'");     // CHANGE}rclcpp::Subscription<tutorial_interfaces::msg::Num>::SharedPtr subscription_;  // CHANGE
};int main(int argc, char * argv[])
{rclcpp::init(argc, argv);rclcpp::spin(std::make_shared<MinimalSubscriber>());rclcpp::shutdown();return 0;
}

调整改变的地方也是与publisher差不多,都是将之前的std_msgs::msg::String消息换成了tutorial_interfaces::msg::Num消息。

CMakeLists.txt

CMakeLists.txt中将std_msgs替换为新包tutorial_interfaces即可,

#...find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(tutorial_interfaces REQUIRED)                      # CHANGEadd_executable(talker src/publisher_member_function.cpp)
ament_target_dependencies(talker rclcpp tutorial_interfaces)    # CHANGEadd_executable(listener src/subscriber_member_function.cpp)
ament_target_dependencies(listener rclcpp tutorial_interfaces)  # CHANGEinstall(TARGETStalkerlistenerDESTINATION lib/${PROJECT_NAME})ament_package()

package.xml

只需修改一行即可,如下:

<depend>tutorial_interfaces</depend>

构建包cpp_pubsub

进入工作空间根路径,执行下面命令进行构建:

$colcon build --packages-select cpp_pubsub

分别新开两个终端,各自source install/setup.bash,再运行订阅者节点再发布者节点,结果如下:

AddThreeInts.srv 

        测试新定义的服务,我们同样可以修改之前的例子(服务端/客户端),进入cpp_srvcli包,先修改add_two_ints_server.cpp:

#include "rclcpp/rclcpp.hpp"
#include "tutorial_interfaces/srv/add_three_ints.hpp"                                        // CHANGE#include <memory>void add(const std::shared_ptr<tutorial_interfaces::srv::AddThreeInts::Request> request,     // CHANGEstd::shared_ptr<tutorial_interfaces::srv::AddThreeInts::Response>       response)  // CHANGE
{response->sum = request->a + request->b + request->c;                                      // CHANGERCLCPP_INFO(rclcpp::get_logger("rclcpp"), "Incoming request\na: %ld" " b: %ld" " c: %ld",  // CHANGErequest->a, request->b, request->c);                                         // CHANGERCLCPP_INFO(rclcpp::get_logger("rclcpp"), "sending back response: [%ld]", (long int)response->sum);
}int main(int argc, char **argv)
{rclcpp::init(argc, argv);std::shared_ptr<rclcpp::Node> node = rclcpp::Node::make_shared("add_three_ints_server");   // CHANGErclcpp::Service<tutorial_interfaces::srv::AddThreeInts>::SharedPtr service =               // CHANGEnode->create_service<tutorial_interfaces::srv::AddThreeInts>("add_three_ints",  &add);   // CHANGERCLCPP_INFO(rclcpp::get_logger("rclcpp"), "Ready to add three ints.");                     // CHANGErclcpp::spin(node);rclcpp::shutdown();
}

再修改add_two_ints_client.cpp:

#include "rclcpp/rclcpp.hpp"
#include "tutorial_interfaces/srv/add_three_ints.hpp"                                       // CHANGE#include <chrono>
#include <cstdlib>
#include <memory>using namespace std::chrono_literals;int main(int argc, char **argv)
{rclcpp::init(argc, argv);if (argc != 4) { // CHANGERCLCPP_INFO(rclcpp::get_logger("rclcpp"), "usage: add_three_ints_client X Y Z");      // CHANGEreturn 1;}std::shared_ptr<rclcpp::Node> node = rclcpp::Node::make_shared("add_three_ints_client");  // CHANGErclcpp::Client<tutorial_interfaces::srv::AddThreeInts>::SharedPtr client =                // CHANGEnode->create_client<tutorial_interfaces::srv::AddThreeInts>("add_three_ints");          // CHANGEauto request = std::make_shared<tutorial_interfaces::srv::AddThreeInts::Request>();       // CHANGErequest->a = atoll(argv[1]);request->b = atoll(argv[2]);request->c = atoll(argv[3]);                                                              // CHANGEwhile (!client->wait_for_service(1s)) {if (!rclcpp::ok()) {RCLCPP_ERROR(rclcpp::get_logger("rclcpp"), "Interrupted while waiting for the service. Exiting.");return 0;}RCLCPP_INFO(rclcpp::get_logger("rclcpp"), "service not available, waiting again...");}auto result = client->async_send_request(request);// Wait for the result.if (rclcpp::spin_until_future_complete(node, result) ==rclcpp::FutureReturnCode::SUCCESS){RCLCPP_INFO(rclcpp::get_logger("rclcpp"), "Sum: %ld", result.get()->sum);} else {RCLCPP_ERROR(rclcpp::get_logger("rclcpp"), "Failed to call service add_three_ints");    // CHANGE}rclcpp::shutdown();return 0;
}

由之前请求两个元素变成请求三个元素,不再赘述。

CMakeLists.txt

将std_msgs替换为新定义的tutorial_interfaces,

#...find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(tutorial_interfaces REQUIRED)         # CHANGEadd_executable(server src/add_two_ints_server.cpp)
ament_target_dependencies(serverrclcpp tutorial_interfaces)                      # CHANGEadd_executable(client src/add_two_ints_client.cpp)
ament_target_dependencies(clientrclcpp tutorial_interfaces)                      # CHANGEinstall(TARGETSserverclientDESTINATION lib/${PROJECT_NAME})ament_package()

package.xml

同样只需修改这一行

<depend>tutorial_interfaces</depend>

构建包cpp_srvcli

工作空间根路径下:

$colcon build --packages-select cpp_srvcli

同样开启两个终端,分别source install/setup.bash,然后先启动服务节点,然后启动客户端节点,结果如下:

 

从以上结果可以看出,我们的自定义msg和srv成功了!

本篇完。

这篇关于ROS 2边学边练(16)-- 自定义msg和srv文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

Druid连接池实现自定义数据库密码加解密功能

《Druid连接池实现自定义数据库密码加解密功能》在现代应用开发中,数据安全是至关重要的,本文将介绍如何在​​Druid​​连接池中实现自定义的数据库密码加解密功能,有需要的小伙伴可以参考一下... 目录1. 环境准备2. 密码加密算法的选择3. 自定义 ​​DruidDataSource​​ 的密码解密3

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

如何自定义Nginx JSON日志格式配置

《如何自定义NginxJSON日志格式配置》Nginx作为最流行的Web服务器之一,其灵活的日志配置能力允许我们根据需求定制日志格式,本文将详细介绍如何配置Nginx以JSON格式记录访问日志,这种... 目录前言为什么选择jsON格式日志?配置步骤详解1. 安装Nginx服务2. 自定义JSON日志格式各

Android自定义Scrollbar的两种实现方式

《Android自定义Scrollbar的两种实现方式》本文介绍两种实现自定义滚动条的方法,分别通过ItemDecoration方案和独立View方案实现滚动条定制化,文章通过代码示例讲解的非常详细,... 目录方案一:ItemDecoration实现(推荐用于RecyclerView)实现原理完整代码实现

基于Spring实现自定义错误信息返回详解

《基于Spring实现自定义错误信息返回详解》这篇文章主要为大家详细介绍了如何基于Spring实现自定义错误信息返回效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录背景目标实现产出背景Spring 提供了 @RestConChina编程trollerAdvice 用来实现 HTT

SpringSecurity 认证、注销、权限控制功能(注销、记住密码、自定义登入页)

《SpringSecurity认证、注销、权限控制功能(注销、记住密码、自定义登入页)》SpringSecurity是一个强大的Java框架,用于保护应用程序的安全性,它提供了一套全面的安全解决方案... 目录简介认识Spring Security“认证”(Authentication)“授权” (Auth

SpringBoot自定义注解如何解决公共字段填充问题

《SpringBoot自定义注解如何解决公共字段填充问题》本文介绍了在系统开发中,如何使用AOP切面编程实现公共字段自动填充的功能,从而简化代码,通过自定义注解和切面类,可以统一处理创建时间和修改时间... 目录1.1 问题分析1.2 实现思路1.3 代码开发1.3.1 步骤一1.3.2 步骤二1.3.3

dubbo3 filter(过滤器)如何自定义过滤器

《dubbo3filter(过滤器)如何自定义过滤器》dubbo3filter(过滤器)类似于javaweb中的filter和springmvc中的intercaptor,用于在请求发送前或到达前进... 目录dubbo3 filter(过滤器)简介dubbo 过滤器运行时机自定义 filter第一种 @A