【手把手做ROS2机器人系统开发五】使用C++实现编写简单的服务器和客户端

该文详细介绍了如何在ROS2环境下使用C++编写服务器和客户端程序,包括创建软件包、编译、配置、服务器端和客户端代码实现,以及程序测试的过程。服务器端实现两个数相加的服务,客户端则向服务器发送请求并接收响应。

使用C++实现编写简单的服务器和客户端

目录

使用C++实现编写简单的服务器和客户端

一、程序编写

1、创建软件包

 2、编译软件包

3、软件配置

4、服务器程序编写

5、客户端程序编写

6、软件包设置

7、设置编译选项

二、程序测试

1、编译程序

2、开启节点测试运行

3、执行效果展示


        上一讲我们讲解了如何编写一个发布者和订阅者程序。本章节继续讲解程序模板,如何编写一个服务器和客户端程序。对于后续的实战开发,前期的理论教学还是少不了,请大家多多阅读。

一、程序编写

1、创建软件包

        继续将软件包创建src目录下,方便统一存放和管理。

ros2 pkg create --build-type ament_cmake cpp_srvcli --dependencies rclcpp example_interfaces

 2、编译软件包

        回到上级目录进行编译工作,注意路径。

colcon build --packages-select cpp_srvcli

3、软件配置

 软件编程环境配置,参考上一章节

【手把手做ROS2机器人系统开发三】搭建vscode编程环境

【手把手做ROS2机器人系统开发四】使用C++实现编写简单的发布者和订阅者

4、服务器程序编写

服务器代码路径:

src/cpp_srvcli/src/add_two_ints_server.cpp

实现两个数相加基本运算服务

/**
 * @file add_two_ints_server.cpp
 * @author gmotion (motion_gui@126.com)
 * @brief 发布服务器节点
 * @version 0.1
 * @date 2022-06-10
 * @copyright Copyright (c) 2022
 */

#include <memory>       //内存管理
#include <string>       //字符串

#include <rclcpp/rclcpp.hpp>
//这里使用示例中的接口,后续章节会重点讲解如何自定义专用接口
#include <example_interfaces/srv/add_two_ints.hpp>

//添加服务信息
void add(const std::shared_ptr<example_interfaces::srv::AddTwoInts::Request>reques,
        std::shared_ptr<example_interfaces::srv::AddTwoInts::Response>response)
{
    response->sum = reques->a + reques->b;
    RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"Incoming request\n a:%ld " "b: %ld",
    reques->a,reques->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();
    return 0;
}

5、客户端程序编写

客户端代码路径:

src/cpp_srvcli/src/add_two_ints_client.cpp

实现请求服务器调用计算,测试服务器是否正确运行

/**
 * @file add_two_ints_client.cpp
 * @author gmotion (motion_gui@126.com)
 * @brief 客户端-请求服务器计算
 * @version 0.1
 * @date 2022-06-10
 * @copyright Copyright (c) 2022
 */

#include <chrono>       //处理时间类
#include <cstdlib>      //标准函数库
#include <memory>       //内存管理

#include <rclcpp/rclcpp.hpp>
#include <example_interfaces/srv/add_two_ints.hpp>

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;
    }

    //建立node节点
    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_INFO(rclcpp::get_logger("rclcpp"),"service no avaiable,wait again...");
        }
    }
    
    //发送运行请求
    auto result = client->async_send_request(request);
    //等待运行结果
    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_two_ints");
    }

    rclcpp::shutdown();
    return 0;
}

6、软件包设置

src/cpp_srvcli/package.xml
<?xml version="1.0"?>
<?xml-model href="http://download.ros.org/schema/package_format3.xsd" schematypens="http://www.w3.org/2001/XMLSchema"?>
<package format="3">
  <name>cpp_srvcli</name>
  <version>0.0.0</version>
  <description>C++ client server tutorial</description>
  <maintainer email="motion_gui@126.com">gmotion</maintainer>
  <license>Apache License 2.0</license>

  <buildtool_depend>ament_cmake</buildtool_depend>

  <depend>rclcpp</depend>
  <depend>example_interfaces</depend>

  <test_depend>ament_lint_auto</test_depend>
  <test_depend>ament_lint_common</test_depend>

  <export>
    <build_type>ament_cmake</build_type>
  </export>
</package>

7、设置编译选项

src/cpp_srvcli/CMakeLists.txt
cmake_minimum_required(VERSION 3.8)
project(cpp_pubsub)

# Default to C99
if(NOT CMAKE_C_STANDARD)
  set(CMAKE_C_STANDARD 99)
endif()

# Default to C++14
if(NOT CMAKE_CXX_STANDARD)
  set(CMAKE_CXX_STANDARD 14)
endif()


if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
  add_compile_options(-Wall -Wextra -Wpedantic)
endif()

# find dependencies
find_package(ament_cmake REQUIRED)
# uncomment the following section in order to fill in
# further dependencies manually.
# find_package(<dependency> REQUIRED)
find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)


add_executable(talker src/publisher_member_function.cpp)
ament_target_dependencies(talker rclcpp std_msgs)

add_executable(listener src/subscriber_member_function.cpp)
ament_target_dependencies(listener rclcpp std_msgs)

install(TARGETS
  talker
  listener
  DESTINATION lib/${PROJECT_NAME})


if(BUILD_TESTING)
  find_package(ament_lint_auto REQUIRED)
  # the following line skips the linter which checks for copyrights
  # comment the line when a copyright and license is added to all source files
  set(ament_cmake_copyright_FOUND TRUE)
  # the following line skips cpplint (only works in a git repo)
  # comment the line when this package is in a git repo and when
  # a copyright and license is added to all source files
  set(ament_cmake_cpplint_FOUND TRUE)
  ament_lint_auto_find_test_dependencies()
endif()

ament_package()

二、程序测试

1、编译程序

单一软件包指定编译

colcon build --packages-select cpp_srvcli

2、开启节点测试运行

服务器终端:

. install/setup.bash
ros2 run cpp_srvcli server

客户端终端:

. install/setup.bash
ros2 run cpp_srvcli client 12 10

3、执行效果展示

         程序准确执行,达到预期结果。本节实现一个简单的服务器节点和一个客户端节点。更多精彩内容,欢迎订阅,敬请阅读下一章内容。谢谢大家的阅读。。。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值