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

相关文章

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

Oracle type (自定义类型的使用)

oracle - type   type定义: oracle中自定义数据类型 oracle中有基本的数据类型,如number,varchar2,date,numeric,float....但有时候我们需要特殊的格式, 如将name定义为(firstname,lastname)的形式,我们想把这个作为一个表的一列看待,这时候就要我们自己定义一个数据类型 格式 :create or repla

【JavaScript】LeetCode:16-20

文章目录 16 无重复字符的最长字串17 找到字符串中所有字母异位词18 和为K的子数组19 滑动窗口最大值20 最小覆盖字串 16 无重复字符的最长字串 滑动窗口 + 哈希表这里用哈希集合Set()实现。左指针i,右指针j,从头遍历数组,若j指针指向的元素不在set中,则加入该元素,否则更新结果res,删除集合中i指针指向的元素,进入下一轮循环。 /*** @param

HTML5自定义属性对象Dataset

原文转自HTML5自定义属性对象Dataset简介 一、html5 自定义属性介绍 之前翻译的“你必须知道的28个HTML5特征、窍门和技术”一文中对于HTML5中自定义合法属性data-已经做过些介绍,就是在HTML5中我们可以使用data-前缀设置我们需要的自定义属性,来进行一些数据的存放,例如我们要在一个文字按钮上存放相对应的id: <a href="javascript:" d

一步一步将PlantUML类图导出为自定义格式的XMI文件

一步一步将PlantUML类图导出为自定义格式的XMI文件 说明: 首次发表日期:2024-09-08PlantUML官网: https://plantuml.com/zh/PlantUML命令行文档: https://plantuml.com/zh/command-line#6a26f548831e6a8cPlantUML XMI文档: https://plantuml.com/zh/xmi

ROS - C++实现RosBag包回放/提取

文章目录 1. 回放原理2. 回放/提取 多个话题3. 回放/提取数据包,并实时发布 1. 回放原理 #include <ros/ros.h>#include <rosbag/bag.h>#include <std_msgs/String.h>int main(int argc, char** argv){// 初始化ROS节点ros::init(argc, argv,

argodb自定义函数读取hdfs文件的注意点,避免FileSystem已关闭异常

一、问题描述 一位同学反馈,他写的argo存过中调用了一个自定义函数,函数会加载hdfs上的一个文件,但有些节点会报FileSystem closed异常,同时有时任务会成功,有时会失败。 二、问题分析 argodb的计算引擎是基于spark的定制化引擎,对于自定义函数的调用跟hive on spark的是一致的。udf要通过反射生成实例,然后迭代调用evaluate。通过代码分析,udf在