之前讲的话题通信是基于订阅/发布机制的,无论有没有订阅者,发布者都会周期发布数据,这种模式适合持续数据的收发,比如传感器数据。机器人系统中还有另外一些配置性质的数据,并不需要周期处理,此时就要用到另外一种ROS通信方式——服务(Service)。 服务是基于客户端/服务器模型的通信机制,服务器端只在接收到客户端请求时才会提供反馈数据。
接下来还是以海龟仿真为例,看看里边有哪些服务的应用呢?
1.启动海龟仿真器
还是用这两个命令来启动海龟仿真器:
ros2 run turtlesim turtlesim_node
ros2 run turtlesim turtle_teleop_key
2.查看服务列表
在小海龟背后,有哪些服务呢?可以使用“ros2 service list”命令来查看一下: 在列表中可以看到每个节点都有6个以“parameters”结尾的服务,在ROS2中,几乎每一个节点都会有这几个服务,主要是用来设置一些参数的,算是节点的默认服务配置,这里我们暂时忽略这些默认的服务,主要看其他几个海龟仿真器提供的功能性质的服务。
3.ros2服务类型
类似话题中的消息,服务中完成通信的数据也是有数据结构的,只不过是有请求和应答两部分的数据组成的。我们可以通过以下命令来查看某一个服务的数据结构:
ros2 service type <service_name>
以/clear服务为例,可以这样来试试:
ros2 service type /clear
看到的服务数据结构是这样的: Empty类型表示服务请求部分的数据是没有的,发送请求的时候不需要任何数据。
3.1 ros2 service list -t
类似查看所有节点的话题消息类型,也可以用类似的方式查看所有服务的数据类型,需要在list命令后边加上“–show-types”配置,或者简写为“-t”。
4.查找提供某类型数据的所有服务
ROS2还允许查看所有提供同样数据类型的的服务,可以使用如下命令:
ros2 service find <type_name>
比如可以查找所有提供std_srvs/srv/Empty数据类型的服务:
ros2 service find std_srvs/srv/Empty
5.查看服务数据类型的具体结构
现在我们已经可以看到某一个服务的数据类型了,那这种数据类型具体的数据结构是什么样的呢?可以用这个命令:
ros2 interface show <type_name>.srv
比如:
ros2 interface show std_srvs/srv/Empty.srv
终端中会显示: 这里的“—”在服务的数据结构中是用来分割请求和应答两个部分的数据,这里只所以只有“—”,是应为Empty的请求和应答都不需要任何数据描述,类似一个出发信号。 我们来看另外一个服务的数据类型Spawn,之前已经根据list命令看到该服务的数据类型是 turtlesim/srv/Spawn:
ros2 interface show turtlesim/srv/Spawn.srv
终端中可以看到: 这里可以看到,在Spawn服务的请求部分,由必须的x、y、theta和可选的name组成,应答数据则是name这个字符串。
6.通过终端发送服务请求
我们已经了解了如何查看服务的数据类型和结构,接下来通过终端尝试发送一个服务请求。
ros2 service call <service_name> <service_type> <arguments>
<arguments>部分是可选的,比如Empty类型的服务就没有任何参数:
ros2 service call /clear std_srvs/srv/Empty
这个服务只是出发一个信号,让海龟仿真器清除海龟后边的运动轨迹,不需要传输什么数据: 我们再来试试Spawn服务,这个服务可以产生一只新的海龟,<arguments>部分就需要输入上边看到的请求数据了:
ros2 service call /spawn turtlesim/srv/Spawn "{x: 2, y: 2, theta: 0.2, name: ''}"
终端中可以看到: 海龟仿真器中立刻就可以看到一只新的海龟啦: 请求数据中的x、y、theta表示海龟的横纵坐标和旋转角度,name表示新生海龟的名字。 好啦,这就是ROS2中的服务概念啦,简而言之,客户端发送请求,服务端完成处理后反馈应答,通信只会交互一次数据,不像话题是周期发送数据的。