ROS2-service通信及闪退的问题

1. 简介

service通信和topic通信的主要区别在于service是有应答的,使用起来很简单,在网上一搜就有例程,之所以整理这篇笔记是因为本人在使用中遇到一个闪退的问题,排查了好一阵子才解决,为了后人少采坑整理本笔记

本笔记内容如下:

  • 简单介绍service在ros2使用
    • 客户端
    • 服务端
  • spin的闪退问题
  • 常用命令

2. service在ros2使用

2.1 服务端

不多说直接copy例程

#include "rclcpp/rclcpp.hpp"
#include "example_interfaces/srv/add_two_ints.hpp"

#include <memory>

void add(const std::shared_ptr<example_interfaces::srv::AddTwoInts::Request> request,
          std::shared_ptr<example_interfaces::srv::AddTwoInts::Response>      response)
{
  response->sum = request->a + request->b;
  RCLCPP_INFO(rclcpp::get_logger("rclcpp"), "Incoming request\na: %ld" " b: %ld",
                request->a, request->b);
  RCLCPP_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_two_ints_server");

  rclcpp::Service<example_interfaces::srv::AddTwoInts>::SharedPtr service =
    node->create_service<example_interfaces::srv::AddTwoInts>("add_two_ints", &add);

  RCLCPP_INFO(rclcpp::get_logger("rclcpp"), "Ready to add two ints.");

  rclcpp::spin(node);
  rclcpp::shutdown();
}

关键点:

  • create_service创建server
    • 需要注明服务端的名称,客户端是通过该名字匹配
    • 提供回调处理函数

2.2 客户端

#include "rclcpp/rclcpp.hpp"
#include "example_interfaces/srv/add_two_ints.hpp"

#include <chrono>
#include <cstdlib>
#include <memory>

using namespace std::chrono_literals;

int main(int argc, char **argv)
{
  rclcpp::init(argc, argv);

  if (argc != 3) {
      RCLCPP_INFO(rclcpp::get_logger("rclcpp"), "usage: add_two_ints_client X Y");
      return 1;
  }

  std::shared_ptr<rclcpp::Node> node = rclcpp::Node::make_shared("add_two_ints_client");
  rclcpp::Client<example_interfaces::srv::AddTwoInts>::SharedPtr client =
    node->create_client<example_interfaces::srv::AddTwoInts>("add_two_ints");

  auto request = std::make_shared<example_interfaces::srv::AddTwoInts::Request>();
  request->a = atoll(argv[1]);
  request->b = atoll(argv[2]);

  while (!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::executor::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_two_ints");
  }

  rclcpp::shutdown();
  return 0;
}

关键点:

  • create_client创建客户端对象,需要指明服务端的名字;
  • wait_for_service检查服务端是否存在;
  • std::make_shared<SRV_MSG::Request>创建请求消息;
  • async_send_request异步发送请求:这个API的返回是一个std::shared_future对象,简单来说这类对象是C++11的std库提供用于异步操作,该对象用于等待"未发生"的事件;
  • spin_until_future_complete用于检查刚才返回的"未发生"对象,输入参数可以增加超时时间

3. 闪退问题

3.1 现象描述

在应用上面的客户端时发现,程序执行到spin_until_future_complete这个api时出现闪退.

3.2 分析

刚好留意到这个api带有spin字样,以及输入node节点,这个和我们使用的rclcpp::spin(pNode),是不是同一个节点不能调两个spin呢?

刚好找到以下的博客遇到同样的问题
https://fishros.org.cn/forum/topic/421/ros2自定义客户端中spin_until_future_complete的问题/2

我想将client封装为一个类,类中定义一个定时器以及一个客户端,定时器回调函数中会发送客户端请求,同时阻塞直到接受server的response(使用spin_until_future_complete方法)。这样有一个问题,rclcpp::spin_until_future_complete( )相当于执行了一次spin,而我定义的类,最终是要添加到executors中,executors再调用spin()方法,本质上就是spin中又调用了spin,在运行时会出现冲突。

3.3 目的和意义

这个用法其实是常见的:

  • 一种就是和博客上面的情况,用ros的定时器设了回调,回调里面再执行轮询服务端,定时器的回调肯定是通过另一个spin来拉起的(关于ros的spin工作原理请翻看另一篇笔记);
  • 另一种就是结点里面注册了一些话题的回调,同样必然需要放入spin,然后在主线程中执行和服务端的通讯,然后阻塞等待结果返回

所以这种问题还是很容易遇到的.

3.4 解决方法

方法:阻塞等待时不使用ros的spin_until_future_complete,而使用std::shared_future自身的wait_for,如下:

auto response = camera_switchers_->async_send_request(request);

auto status = response.wait_for(100ms);

// Wait for the result.
if(status == std::future_status::timeout) {
    return response.get()->result;
} else {
    RCLCPP_ERROR(p_node_->get_logger(), "Fail to call server.");
    return false;
}

3.5 补充

另外上面提到的博客他还遇到另一个问题,通过他的描述我估计是这样的:
因为它的阻塞等待放在回调中,而wait_for本身又是要等待服务应答后触发另一种回调,这时候如果不设置可重入回调组会导致wait_for返回的永远是超时,这又涉及到ros的spin机制,详细看另一篇笔记.

4. 常用指令

  1. ros2 service call
ros2 service call [service name] [msg type] [content]

# 例子
ros2 service call /camera_control_service \
keystar_msgs/srv/SwitchCamera \ 
"{node_name: 'keystar',arm_id: 0,camera_id: 0,enable: true}"
  1. ros2 interface show

和topic一样用于查看服务消息接口的定义

  1. ros2 service list \[-t]

-t 选项指标记消息类型

  • 15
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ROS2 Web Bridge是一个开源的软件包,旨在使ROS2(Robot Operating System 2)与Web端进行通信和交互。它提供了一种简单而强大的方式,通过WebSocket协议将ROS2系统中的数据传输到Web浏览器。 ROS2 Web Bridge允许Web开发人员使用常见的Web技术(例如JavaScript)直接与ROS2系统进行交互。它提供了一个轻量级的接口,可以订阅和发布ROS2主题,并在Web浏览器中实时显示传感器数据、控制机器人等。 此外,ROS2 Web Bridge还支持ROS2服务和动作。它允许Web应用程序在Web端调用ROS2服务,从而实现与ROS2节点的双向通信。通过一个用户友好的Web界面,用户可以发送控制命令给机器人,执行任务并获得实时反馈。 ROS2 Web Bridge的主要优点之一是其跨平台性。它基于WebSocket协议,因此可以在不同的操作系统和设备上使用,无论是在PC端还是移动设备上。 此外,ROS2 Web Bridge还支持认证和授权机制,以确保通信安全。这对于确保只有被授权的用户可以访问和控制ROS2系统非常重要。 总的来说,ROS2 Web Bridge为ROS2系统提供了一种简便而强大的方式,使Web开发人员能够与ROS2系统进行交互。它扩展了ROS2的功能,使得机器人开发更加灵活和可视化。 ### 回答2: ros2-web-bridge是一种用于在ROS 2和Web应用程序之间进行通信的桥接工具。ROS 2是机器人操作系统的第二代版本,而Web应用程序是通过Web浏览器访问的应用程序。 ros2-web-bridge有几个主要功能。首先,它允许ROS 2中的节点直接与通过Web浏览器访问的Web应用程序进行通信。这使得在Web界面上实时监控和控制ROS 2系统变得更加容易。例如,可以使用ros2-web-bridge将传感器数据从ROS 2节点发送到Web应用程序,以便在Web界面上实时显示传感器数据。同时,还可以将来自Web界面的用户输入发送到ROS 2节点,以便远程控制机器人或系统。 其次,ros2-web-bridge还提供了一些工具和API,用于将ROS 2中的消息和服务转换为Web格式。这使得可以轻松地在ROS 2和Web应用程序之间进行数据传输和交互。它支持使用ROS 2的套接字和JSON进行通信,并提供了将消息和服务转换为JSON格式以及反向转换的功能。这样,ROS 2节点可以与通过Web浏览器访问的Web应用程序进行无缝通信。 总而言之,ros2-web-bridge为ROS 2和Web应用程序之间的通信提供了一个便捷的桥梁。它简化了ROS 2系统与Web界面之间的集成,并提供了实时数据传输和远程控制的能力。这对于构建基于ROS 2的机器人或系统的开发者和使用者来说是非常方便的工具。 ### 回答3: ros2-web-bridge是一个用于将ROS 2和Web技术进行连接的工具。它提供了一个桥接器,使得可以通过Web浏览器与ROS 2通信。这个工具是建立在ROS 2和Web Socket之间的通信基础上的。 通过ros2-web-bridge,我们可以在Web浏览器中实时地订阅和发布ROS 2的消息。这使得我们可以通过Web界面来控制ROS 2的机器人,或者将ROS 2的数据可视化展示出来。这对于远程监控、远程操作和数据可视化都非常有用。 ros2-web-bridge使用ROS 2提供的接口来与ROS 2系统进行通信。它将ROS 2的消息转换为适用于Web Socket的格式,并在浏览器和ROS 2之间建立起适配的连接。通过这种方式,Web界面就能够与ROS 2系统进行实时的双向通信ROS 2中的消息传递方式是异步的,而Web浏览器通常使用同步的方式进行通信ros2-web-bridge通过在ROS 2和Web Socket之间进行适配和转换,使得二者能够协同工作。这意味着ROS 2中的数据可以通过ros2-web-bridge传输到Web浏览器,并在该浏览器中进行处理和展示。 总的来说,ros2-web-bridge是一个功能强大的工具,它架起了ROS 2和Web技术之间的桥梁。它使得我们可以通过Web浏览器来与ROS 2进行通信,进而实现远程操作和数据可视化的目的。通过ros2-web-bridge,我们可以更加灵活和方便地利用ROS 2的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值