ROS | 服务通信的编程实现

2023-05-16

ROS | 服务通信的编程实现

  • 1. 创建功能包
  • 2. 节点编程与服务数据定义
    • 2.1 案例说明
    • 2.2 服务数据的定义
    • 2.3 创建.cpp文件
    • 2.4 客户端编程
    • 2.5 服务器编程
  • 3. 配置与编译
    • 3.1 在CMaKeLists.txt中添加编译选项
    • 3.2 在package.xml中添加功能包依赖
    • 3.3 编译文件

1. 创建功能包

在ROS工作空间ROS_ws的src文件夹目录下创建一个功能包,命名为communication_pkg,并编译完成。
在这里插入图片描述

2. 节点编程与服务数据定义

2.1 案例说明

定义一个客户端Client,通过自定义的服务请求Request将两个整数发送给服务器Server,同时服务器Server将两个整数相加后,再通过自定义的服务应答Response将两个整数的和反馈到客户端Client。
在这里插入图片描述

2.2 服务数据的定义

在功能包目录下创建一个新的文件夹,命名为srv,并在此文件夹中创建一个空文件AddTwoInts.srv。
在这里插入图片描述
在AddTwoInts.srv文件中输入以下代码,定义服务消息内容。

int64 a
int64 b
---
int64 sum

说明:

  • 使用三个小横杠“—”作为分隔符,“—”上面代表服务请求的消息内容,“—”下面代表服务应答的消息内容。附相关资料:https://wiki.ros.org/srv

2.3 创建.cpp文件

在功能包下面的src文件夹目录下创建一个空文件client.cpp。

2.4 客户端编程

打开上面所创建的文件client.cpp,输入以下代码。

#include <cstdlib>
#include "ros/ros.h"
#include "communication_pkg/AddTwoInts.h"
	
int main(int argc, char **argv)
{
	//初始化ROS节点
	ros::init(argc, argv, "add_two_ints_client");
	
	//从终端获取数据
	if(argc != 3)
	{
		ROS_INFO("usage: add_two_ints_client X Y");
		return 1;
	}
	
	//创建句柄
	ros::NodeHandle n;
	
	//创建一个客户端,定义服务请求数据类型communication_pkg::AddTwoInts
	ros::ServiceClient client = n.serviceClient<communication_pkg::AddTwoInts>("add_two_ints");
	
	//定义服务消息内容
	communication_pkg::AddTwoInts srv;
	srv.request.a = atoll(argv[1]);
	srv.request.b = atoll(argv[2]);
	
	//发布服务请求,等待应答结果
	if(client.call(srv))
	{
		ROS_INFO("Sum: %ld", (long int)srv.response.sum);
	}
	else
	{
		ROS_ERROR("Failed to call service add_two_ints");
		return 1;
	}
	
	return 0;
}

说明:

  • 头文件cstdlib.h是C++里面的一个常用函数库,等价于C中的stdlib.h,可以提供一些函数与符号常量。
  • 头文件ros/ros.h包含了标准ROS类的声明,在每一个ROS程序中都需要包含它。
  • 头文件communication_pkg/AddTwoInts.h是由AddTwoInts.srv编译扩展得到,包含了针对C++类的定义等,它存放在工作空间的devel/include路径下的communication_pkg文件夹中。
  • int main(int argc, char **argv)是main函数的一种定义形式,其参数argc和argv用于运行时,把命令行参数传入主程序,其中arg是指arguments,即参数。
    • int argc:英文名为arguments count(参数计数),表示运行程序传送给main函数的命令行参数总个数,包括可执行程序名,其中当argc=1时表示只有一个程序名称,此时存储在argv[0]中。
    • char **argv:英文名为arguments value/vector(参数值),用来存放指向字符串参数的指针数组,每个元素指向一个参数,空格分隔参数,其长度为argc。数组下标从0开始,argv[argc]=NULL。argv[0]指向程序运行时的全路名;argv[1]指向程序在DOS命令中执行程序名后的第一个字符串;argv[2]指向执行程序名后的第二个字符串。
  • ros::init(argc, argv, “add_two_ints_client”)的作用是初始化ROS节点,第三个参数表示节点名称,这个节点名是唯一的。
  • if(argc != 3){…}表示当终端命令行输入的参数数量不等于3,则执行这段代码。
  • ros::NodeHandle n;的作用是创建句柄,启动ROS节点。
  • ros::ServiceClient client = n.serviceClient<communication_pkg::AddTwoInts>(“add_two_ints”);这是用户调用ROS服务的其中一种方法。它创建了一个客户端,其返回的ros::ServiceClient类的对象client之后将用于向ROS网络中名为add_two_ints的节点发送服务请求,服务请求的数据类型为communication_pkg::AddTwoInts。
communication_pkg::AddTwoInts srv;
srv.request.a = atoll(argv[1]);
srv.request.b = atoll(argv[2]);
  • 这是创建服务请求和服务应答的其中一种方法。它定义了一个service_example::AddTwoInts类型的对象,该对象中的成员正是srv文件中定义的a、b、sum,将终端命令行输入的两个数填充到数据成员a、b中。atoll()函数的作用是将字符串转换成长长整型数(long long int),即srv.request.a = atoll(argv[1])是将命令行输入的字符串中的第一个数转换成长长整型数,并赋值给成员a,srv.request.b = atoll(argv[2])则同理。
  • if(client.call(srv)){…}表示向ROS网络发起服务请求。由于服务请求处于阻塞状态,需要等待请求发起完成后才能返回值。如果服务调用成功,则call()将返回true,并且srv.response中的值将有效。如果调用失败,则call()将返回false,并且srv.response中的值将无效。
  • ROS_INFO()和ROS_ERROR()用于输出日志信息,日志消息分为五个不同的严重级别,按照严重性程度递增,分别为DEBUG、INFO、WARN、ERROR、FATAL。

综上所述,实现一个客户端的步骤大致可分为以下几点:

  1. 初始化ROS节点;
  2. 创建client实例;
  3. 发布服务请求数据;
  4. 等待Server处理之后的应答结果。

2.5 服务器编程

在src文件夹下再创建一个空文件server.cpp,输入以下代码。

#include "ros/ros.h"
#include "communication_pkg/AddTwoInts.h"
	
//服务回调函数,输入为服务请求Req,输出为服务应答Res
bool add(communication_pkg::AddTwoInts::Request  &req,
		 communication_pkg::AddTwoInts::Response &res)
{
	//将两个数相加,并将结果存放在变量sum中
	res.sum = req.a + req.b;
	ROS_INFO("request: x=%ld, y=%ld", (long int)req.a, (long int)req.b);
	ROS_INFO("sending back response: [%ld]", (long int)res.sum);
	
	return true;
}
	
int main(int argc, char **argv)
{
	//初始化ROS节点
	ros::init(argc, argv, "add_two_ints_server");
	
	//创建句柄
	ros::NodeHandle n;
	
	//创建服务器,登记回调函数
	ros::ServiceServer service = n.advertiseService("add_two_ints", add);
	
	//循环等待
	ROS_INFO("Ready to add two ints.");
	ros::spin();
	
	return 0;
}

说明:

  • main函数中一开始都是类似的,初始化ROS节点,创建句柄,从而启动ROS节点。
  • ros::ServiceServer service = n.advertiseService(“add_two_ints”, add);的作用是创建服务,并将服务加入到ROS网络中。这个服务在ROS网络中以add_two_ints命名,并且作为唯一标识,以便于其他节点通过服务名称进行请求。当接收到服务请求后,则调用add()回调函数。
  • ros::spin()的作用是让程序进入自循环的挂起状态,从而让程序以最好的效率接收客户端的请求并调用回调函数。
  • 服务回调函数的表示形式如下:
bool callback(MReq &request, MRes &response)
  • 其中MReq和MRes与提供给advertiseService()的请求/应答的数据类型相匹配。回调函数的返回值若为true,则表示服务请求成功,并且服务应答已填充必要的数据。若返回值为false则表示服务请求失败,并且服务应答将不会发送给客户端。
  • 回调函数add(communication_pkg::AddTwoInts::Request &req, communication_pkg::AddTwoInts::Response &res)的作用是实现两个int型整数求和的服务,两个整数从request获取,求和结果填充到response里,request与response的具体数据类型在srv文件中被定义。

综上所述,实现一个服务器的步骤大致可分为以下几点:

  1. 初始化ROS节点;
  2. 创建Server实例;
  3. 循环等待服务请求,进入回调函数;
  4. 在回调函数中完成服务功能的处理,并反馈应答数据。

附相关资料:
(1) Writing a Simple Service and Client (C++)
(2) http://wiki.ros.org/roscpp/Overview/Services

3. 配置与编译

3.1 在CMaKeLists.txt中添加编译选项

打开功能包中的CMaKeLists.txt文件,在如下位置的find_package中添加功能包,以便于(节点)调用它们生成消息。
在这里插入图片描述

在如下位置添加相关的.srv文件,确保了CMake在重新配置时知道这些新添加的.srv文件,同时添加.srv文件在生成消息时的所有依赖项(功能包)。

add_service_files(FILES AddTwoInts.srv)

在这里插入图片描述

将如下位置中CATLIN_DEPENDS前面的“#”去掉。
在这里插入图片描述

在如下位置进行配置,add_executable(client src/client.cpp)的作用是将src文件夹下的client.cpp文件编译成名为client的可执行文件。target_link_libraries(client ${catkin_LIBRARIES})的作用是将client可执行文件与ROS相关的库链接。add_dependencies(client ${PROJECT_NAME}_gencpp)的作用是将client可执行文件与一些动态生成的文件链接。

add_executable(server src/server.cpp)
target_link_libraries(server ${catkin_LIBRARIES})
add_dependencies(server ${PROJECT_NAME}_gencpp)
	
add_executable(client src/client.cpp)
target_link_libraries(client ${catkin_LIBRARIES})
add_dependencies(client ${PROJECT_NAME}_gencpp)

在这里插入图片描述

3.2 在package.xml中添加功能包依赖

打开功能包中的package.xml文件,在如下位置添加功能包依赖。

<build_depend>message_generation</build_depend>
<exec_depend>message_runtime</exec_depend>

在这里插入图片描述

说明:

<buildtool_depend>catkin</buildtool_depend>
  • 这条语句表示需要依赖catkin工具包用于使用catkin_make指令进行编译。
<build_export_depend>roscpp</build_export_depend>
  • 这条语句表示编译后导出相关文件需要依赖roscpp功能包。
<build_depend>message_generation</build_depend>
  • 这条语句表示在编译时会依赖一个动态产生message的功能包。
<exec_depend>message_runtime</exec_depend>
  • 这条语句表示在运行时会依赖message_runtime的功能包。

3.3 编译文件

在/ROS_ws文件夹路径下打开一个新的终端,输入以下代码进行编译。

$ catkin_make

在这里插入图片描述

编译完成后,输入以下代码运行主节点。

$ roscore

在这里插入图片描述

打开一个新的终端,配置环境变量后,输入以下代码运行节点。

$ rosrun communication_pkg client 整数1 整数2

在这里插入图片描述

如果rosrun指令后面缺少需要相加的两个数,则会报如下错误。
在这里插入图片描述

打开一个新的终端,配置环境变量后,输入以下代码运行节点。

$ rosrun communication_pkg server

在这里插入图片描述

若想停止运行,关闭终端,使用快捷键Ctrl+c即可

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ROS | 服务通信的编程实现 的相关文章

随机推荐

  • linux gcc编译错误:undefined reference to `aio_error‘解决方法

    include lt aio h gt void aiow completion handler sigval t sigval int ret struct aiocb req req 61 struct aiocb sigval siv
  • wait_event_interruptible_locked的使用方法

    wait event interruptible locked interface New wait event interruptible exclusive locked irq macros added They work just
  • printk在应用层的设置方式及读取内核打印信息的方法

    如果 printk 中没有加调试级别 xff0c 则使用默认的调试级别 注意 xff0c 调试级别和格式化字符串之间没有逗号 当前控制台的各打印级别可以通过下面的命令来查看 cat proc sys kernel printk 4 4 1
  • qt编译Qxlsx模块及安装

    主要参考如下地址 xff1a https www icode9 com content 4 715555 html 注意的点 xff1a 1 把下载的代码复制到根目录下 xff0c 路径不要有什么空格啥的 xff0c 否则你会发现perl老
  • GitLab 使用Tortoisegit询问“git@192.168.1.18‘s password“问题解决

    现象如下 xff1a 使用TortoiseGit去拉本地GitLab上建立的项目时 xff0c 一直提示输入密码 xff08 如下图 xff09 xff0c 这个密码又没有指定用户名 xff0c 就算你输入你用户名的密码也是失败 但是很诡异
  • 二代身份证读写器原理及开发

    身份证读写器的作用就是从身份证中读取身份信息 xff08 例如姓名 民族 身份证号等 xff09 xff0c 然后显示或者传输给其他模块使用 功能框架如下 xff1a 功能框图说明 xff1a 1 业务模块 负责向安全模块发送命令 xff0
  • JLINK V10 V11固件修复

    先去我的资源里面下载bootloader和app固件文件 步骤 xff1a 1 PC上安装JLINK V4 9工具 xff08 貌似不能使用太高版本的工具 xff0c 否则有问题 xff09 2 打开j flash v4 9 xff0c 新
  • ROS | 话题通信的编程实现

    ROS 话题通信的编程实现 1 创建功能包2 节点编程与消息定义2 1 案例说明2 2 话题消息的定义2 3 创建 cpp文件2 4 话题发布者编程2 5 话题订阅者编程 3 配置与编译3 1 在CMaKeLists txt中添加编译选项3
  • Cocos2dx 3.0配置环境

    3 15 cocos2dx 3 0rc0 终于放出来了 在这里不得不吐槽一件事 xff0c 3 0版本从Alpha xff0c 到beta xff0c 再到rc xff0c 三个版本竟然都有各自创建项目的方式 xff0c 这样真的不会被人打
  • linux 开机运行应用程序

    把运行应用程序的脚本放在 etc rc local里面 xff0c 如果没有 etc rc local xff0c 需要执行前面的3条指令新建这个文件 注意执行应用最好要在后台执行 xff08 后面加个 amp xff09 xff0c 否则
  • arm linux游戏手柄(joystick)驱动移植

    在arm linux中移植usb joystick驱动 xff0c 参考了如下经验 xff1a Linux系统中使用Xbox360手柄 知 行 博客园 cnblogs com 使用BlueZ连接蓝牙手柄 Dokin丶的博客 CSDN博客 蓝
  • linux ubuntu下网络调试助手(GUI)工具

    mNetAssist这个工具在ubuntu下可以运行 xff0c 是个带界面的tcp调试工具 1 UDP通讯 xff1b 2 可做 TCP客户端 xff1b 3 可做 TCP服务器 xff1b 4 可以 十六进制 传送接收数据 5 可以传送
  • fft的通俗解释

    FFT是离散傅立叶变换的快速算法 xff0c 可以将一个信号变换 到频域 有些信号在时域上是很难看出什么特征的 xff0c 但是如 果变换到频域之后 xff0c 就很容易看出特征了 这就是很多信号 分析采用FFT变换的原因 另外 xff0c
  • linux 字符驱动完整框架(poll,async,waitqueue,nonblock等)

    一个linux内核驱动的完整框架 xff0c 包含了能遇到的大部分内容 xff0c 例如timer poll async waitqueue nonblock等等 xff0c 不过基本上没啥大用 xff0c 就是用来熟悉基础的 xff0c
  • vscode远程调试Linux CUDA程序

    参考了 xff1a CUDA 01 第一个程序 知乎 zhihu com 1 本地安装插件 xff1a remote ssh xff0c Microsoft C C 43 43 与NVIDIA Nsight Visual Studio Co
  • 移植MQTT-C库(附源码)

    Software mqtt org 中mqtt客户端的c库里面有一个叫MQTT C的库 xff0c 就2个实现文件 xff0c 算比较简单的了 xff0c 实现了基本的mqtt客户端功能 xff0c 移植一下试试 我的移植代码放在我的资源里
  • TCP协议的滑动窗口和流量控制算法(转)

    目录 滑动窗口 流量控制 操作系统缓冲区与滑动窗口的关系 窗口关闭 糊涂窗口综合症 拥塞控制 慢启动 拥塞避免算法 拥塞发生 快速恢复 拥塞算法示意图 引入 窗口概念的原因 我们都知道 TCP 是每发送一个数据 xff0c 都要进行一次确认
  • linux应用中的时间处理

    参考下 xff1a Linux下有关时间的函数 xff1a time times clock gettimeofday等 linux time函数 见牛羊的博客 CSDN博客 下面的代码基本涵盖了获取时间和操作计时的一些函数使用 xff1a
  • 从旋转向量到欧拉角的六种计算方法

    利用SolvePNP解出旋转向量 xff0c 旋转向量通过罗德里格斯公式解出旋转矩阵 xff0c 然后通过下面六种公式计算即可 xff0c 欧拉角有十二种 xff0c 六种是相对于自身参考系 xff0c 六种是相对于惯性参考系 xff0c
  • ROS | 服务通信的编程实现

    ROS 服务通信的编程实现 1 创建功能包2 节点编程与服务数据定义2 1 案例说明2 2 服务数据的定义2 3 创建 cpp文件2 4 客户端编程2 5 服务器编程 3 配置与编译3 1 在CMaKeLists txt中添加编译选项3 2