ROS2中的服务通信

ROS2的服务通信概念与ROS1基本保持一致,服务通信遵循请求响应机制,和ROS1不同的地方在于,ROS2的服务通信不再是同步模式,而是支持异步通信模式(后半部分会有详细介绍)

服务通信的功能示意图

对于服务通信,同样需要创建server和client

  • server创建过程
g_node = rclcpp::Node::make_shared("minimal_service");
auto server = g_node->create_service<AddTwoInts>("add_two_ints", handle_service);
  • client创建过程
auto node = rclcpp::Node::make_shared("minimal_client");
auto client = node->create_client<AddTwoInts>("add_two_ints");

server端,通过回调函数handle_service响应client端的请求

void handle_service(
  const std::shared_ptr<rmw_request_id_t> request_header,
  const std::shared_ptr<AddTwoInts::Request> request,
  const std::shared_ptr<AddTwoInts::Response> response)
{
  (void)request_header;
  RCLCPP_INFO(
    g_node->get_logger(),
    "request: %" PRId64 " + %" PRId64, request->a, request->b);
  response->sum = request->a + request->b;
}

client端检测server端启动成功后,向server端发送请求

 //等待 服务端上线

  while (!client->wait_for_service(std::chrono::seconds(1))) {
    if (!rclcpp::ok()) {
      RCLCPP_ERROR(node->get_logger(), "client interrupted while waiting for service to appear.");
      return 1;
    }
    RCLCPP_INFO(node->get_logger(), "waiting for service to appear...");
  }

//请求 服务
  auto request = std::make_shared<AddTwoInts::Request>();
  request->a = 41;
  request->b = 1;
  auto result_future = client->async_send_request(request);
  if (rclcpp::spin_until_future_complete(node, result_future) !=
    rclcpp::FutureReturnCode::SUCCESS)
  {
    RCLCPP_ERROR(node->get_logger(), "service call failed :(");
    client->remove_pending_request(result_future);
    return 1;
  }
  auto result = result_future.get();
  RCLCPP_INFO(
    node->get_logger(), "result of %" PRId64 " + %" PRId64 " = %" PRId64,
    request->a, request->b, result->sum);
  rclcpp::shutdown();

服务通信的实现逻辑

ROS2中的服务通信是借助消息通信实现的,其流程图如下所示,为了方便说明,图中以上面add_two_ints服务为例:

创建Client时会创建rq/add_two_intsRequest话题,并订阅rr/add_two_intsReply话题,创建Service时会创建rr/add_two_intsReply话题,并订阅rq/add_two_intsRequest话题,这样通过两个话题建立起双向通道,即:Client向rq/add_two_intsRequest话题发消息,Service能收到,Service向rr/add_two_intsReply话题发消息,Client能收到。

Client发送请求消息后,最终调用rmw层的rmw_send_request接口,将请求消息序列化后发送到rq/add_two_intsRequest话题,由于Service订阅了该话题,所以会收到Client的请求消息,并在rmw层的rmw_take_request接口对请求消息进行反序列化,然后将反序列化后的请求消息传给rcl层,由rcl层调用回调函数处理请求消息,得到响应消息,处理结束后,最终调用rmw层的rmw_send_response接口,将响应消息序列化后发送到rr/add_two_intsReply话题,由于Client订阅了该话题,所以会收到Service的响应消息,并在rmw层的rmw_take_response接口对响应消息进行反序列化,然后将反序列化后的响应消息传递给上层,最终,Client拿到Service的响应结果。

临近中秋放假,具体代码调用关系分析,节后再进行.....

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值