ROS通信机制之动作(Action)服务的实践

news2025/1/11 7:15:58

1、动作概述

讲完了 服务 之后,接下来就是通信的第三种机制,动作。在上节我们知道服务的应用场景是需要在有限时间内完成,而对于一些比较复杂的任务,需要耗时比较长,甚至是不确定时间的时候,就需要用到动作这种通信方式了。
在原理上,动作使用 话题 来实现,而且引入了反馈来提供操作的进度信息,也就是说在执行任务的过程中,可以周期性得到反馈,帮助我们了解运行到哪个步骤了,本质上相当于规定了一系列话题(目标,反馈、结果等)的组合使用方法的高层协议。
动作的定义和使用跟服务差不多,稍微复杂一点,但是却非常强大和灵活。

2、动作的定义

创建一个动作,里面的格式分三部分:目标、结果、反馈,同样使用三个短横线进行隔开,跟前面介绍的服务(请求和响应,也就是输入和输出)类似,只不过多了一个反馈消息项,同样的在定义好动作之后进行编译,.action文件也会被打包为一个消息类型。

2.1、.action定义

依然先来看下动作的定义,后缀名是.action,先来到工作区的test包,新建action目录 

cd ~/catkin_ws/src/test
mkdir action
cd action

我们来定义一个定时器Timer.action,进行倒计时,并在倒计时停止时发出信号,在这个过程中将定期告诉我们剩余的时间,计时结束之后,返回总时长。

gedit Timer.action

duration time_to_wait
---
duration time_elapsed
uint32 updates_sent
---
duration time_elapsed
duration time_remaining

目标goal:客户端发送的等待总时长
结果result:服务端发送的等待时长以及更新次数
反馈feedback:服务端周期性反馈,消逝的时长和剩余时长 

加个执行权限:chmod u+x Timer.action

2.2、修改CMakeLists.txt

动作文件定义好了之后,同样需要将编译的相关依赖项给写出来

cd ~/catkin_ws/src/test
gedit CMakeLists.txt

find_package增加actionlib_msgs

find_package(catkin REQUIRED COMPONENTS roscpp rospy std_msgs message_generation actionlib_msgs)

告知catkin需要编译的动作文件

add_action_files(
  DIRECTORY action 
  FILES Timer.action
)

generate_messages显式列出actionlib_msgs消息依赖

generate_messages(DEPENDENCIES actionlib_msgs std_msgs)

catkin_package中添加actionlib_msgs依赖

catkin_package(CATKIN_DEPENDS message_runtime actionlib_msgs) 

2.3、修改package.xml

cd ~/catkin_ws/src/test
gedit package.xml

添加actionlib_msgs

<build_depend>actionlib_msgs</build_depend>
<exec_depend>actionlib_msgs</exec_depend>

2.4、编译

相关准备做好了之后,就对其进行编译,创建该动作实际使用的代码以及类定义:

cd ~/catkin_ws
catkin_make

其中对于编译也可以指定Python版本:

cd ~/catkin_ws
catkin_make -DPYTHON_EXECUTABLE=/usr/bin/python2
cd ~/catkin_ws
catkin_make -DPYTHON_EXECUTABLE=/usr/bin/python3

当然如果在某个版本里面没有安装catkin_pkg将会报错:

ImportError: "from catkin_pkg.package import parse_package" failed: No module named 'catkin_pkg'
Make sure that you have installed "catkin_pkg", it is up to date and on the PYTHONPATH.

比如本人在Python3里面没有安装,重新安装如下:

sudo apt install python3-pip
pip3 install --user catkin_pkg
pip3 install --user empy

查看下,在两个版本下面都安装好了catkin_pkg

sudo find / -name "catkin_pkg"
/usr/lib/python2.7/dist-packages/catkin_pkg
/home/yahboom/.local/lib/python3.6/site-packages/catkin_pkg

编译成功之后,我们来查看下生成了哪些消息文件:

ls ~/catkin_ws/devel/share/test/msg
TimerActionFeedback.msg  TimerAction.msg        TimerFeedback.msg  TimerResult.msg
TimerActionGoal.msg      TimerActionResult.msg  TimerGoal.msg

 这些msg文件里面都是这些动作所定义的类型,我们看下其详细情况,截图如下:

如果指定Python版本去编译,将在对应版本下面,将上面对应的.msg消息文件生成对应的类的定义文件:

cd ~/catkin_ws/devel/lib/python2.7/dist-packages/test/msg
cd ~/catkin_ws/devel/lib/python3/dist-packages/test/msg

ls查看下生成了哪些文件(其中_Complex.py是前面话题章节自定义的消息类型):

_Complex.py   __init__.py   _TimerActionFeedback.py  _TimerAction.py        _TimerFeedback.py  _TimerResult.py
_Complex.pyc  __init__.pyc  _TimerActionGoal.py      _TimerActionResult.py  _TimerGoal.py

从这些生成的目录与文件来看,再次说明了这个动作的本质是话题,而话题本质又是类型的消息流。

3、动作服务器

动作的相关类都已生成好了,接下来就写动作服务器,动作跟话题和服务一样,都使用回调机制,即回调函数会在收到消息时被唤醒和调用。这里我们就直接使用来自actionlib里面的SimpleActionServer类来简化编写过程。
我们只需要定义接收到目标时的回调函数,而回调函数会根据目标来操作定时器,并在操作结束后返回一个结果,除了目标和结果,到后面我们会将反馈再加上,先来看一个基本的动作服务。

cd ~/catkin_ws/src/test/src
gedit action_server1.py
#!/usr/bin/env python
import rospy
import time
import actionlib
from test.msg import TimerGoal,TimerResult,TimerAction

def do_timer(goal):
    start_time = time.time()
    time.sleep(goal.time_to_wait.to_sec())

    result = TimerResult()
    result.time_elapsed = rospy.Duration.from_sec(time.time()-start_time)
    result.updates_sent = 0
    server.set_succeeded(result)

rospy.init_node('timer_action_server')
server = actionlib.SimpleActionServer('timer',TimerAction,do_timer,False)
server.start()
rospy.spin()

添加执行权限:chmod u+x action_server1.py

首先导入相关类以及Timer.action自动生成的类,然后定义回调函数,其中参数goal本质是一个TimerGoal类型的对象,goal中的time_to_wait对象返回的是duration类型,所以需要转成to_sec()秒,反过来需要得到持续时间,这里的time_elapsedDuration类型,使用from_sec()转换。最后就是将结果result作为参数,调用set_succeeded(),告诉SimpleActionServer已成功执行了目标。
我们来看下SimpleActionServer构造函数:

__init__(self, name, ActionSpec, execute_cb=None, auto_start=True)

name:动作服务器的名称
ActionSpec:动作类型
execute_cb:回调函数在单独的线程中调用,接收到一个新的目标,允许用户阻塞回调,添加一个执行回调也会使目标回调失效。
auto_start:告诉ActionServer是否在它出现时立即开始发布,这里应该总是设置为False以出现竞争条件的问题,而导致一些错误的发生,然后显式调用start()来启动动作服务器。那么为什么没有默认为Fasle,这里可能是因为被发现问题时,已经有很多代码都依赖这个默认行为,导致修改变得非常麻烦了。

4、启动检查

动作服务器编写完成之后,我们来检查下是否正常工作。

首先启动:roscore
运行动作服务器:rosrun test action_server1.py
查看话题列表:rostopic list

/rosout
/rosout_agg
/timer/cancel
/timer/feedback
/timer/goal
/timer/result
/timer/status

正确显示出了timer名字空间下的话题,一共有5个,我们来查看一个话题

rostopic info /timer/goal
'''
Type: test/TimerActionGoal

Publishers: None

Subscribers: 
 * /timer_action_server (http://YAB:42723/)

'''

类型是TimerActionGoal,我们来查看下其信息

rosmsg show TimerActionGoal
'''
[test/TimerActionGoal]:
std_msgs/Header header
  uint32 seq
  time stamp
  string frame_id
actionlib_msgs/GoalID goal_id
  time stamp
  string id
test/TimerGoal goal
  duration time_to_wait
'''

在这里也验证了前面说的goalTimerGoal的类型,里面包含了time_to_wait字段,其余部分是服务器和客户端用来追踪动作执行状态的,只不过在目标传入服务器端的回调函数之前,这些信息都将被去除,最后剩下的就是这个TimerGoal消息。我们查看下TimerGoal

rosmsg show TimerGoal
'''
[test/TimerGoal]:
duration time_to_wait
'''

从这些打印的结果来看,如果你使用的是actionlib包中的一些程序库,就不需要去访问这些名字里面含有Action的类型,单纯的Goal、Result、Feedback就完全够用了。

5、动作客户端

服务端我们定义好了,接下来我们定义客户端,向服务器发送一个目标,并且等待结果的到来。

cd ~/catkin_ws/src/test/src
gedit action_client1.py
#!/usr/bin/env python
import rospy
import actionlib
from test.msg import TimerAction,TimerGoal,TimerResult

rospy.init_node('timer_action_client')
client = actionlib.SimpleActionClient('timer',TimerAction)
client.wait_for_server()
goal = TimerGoal()
goal.time_to_wait = rospy.Duration.from_sec(5)
client.send_goal(goal)
client.wait_for_result()
print(client.get_result().time_elapsed.to_sec())

添加可执行权限:chmod u+x action_client1.py

客户端的代码,在导入相关模块之后,创建的SimpleActionClient里面的两个参数跟服务端一样。等待动作服务器启动跟前面的服务rospy.wait_for_service类似,这里是rospy.wait_for_server。接下来就是创建目标,构造一个TimerGoal对象,填入我们想要定时器等待的时间,这里设置为5秒,然后就发送给服务器。最后就是等待服务器的处理结果了,如果一切正常,就打印出获取的结果get_result的持续时间。

运行客户端

rosrun test action_client1.py
#5.005053997

客户端发送5秒的时间给目标,服务端做一个sleep模拟,这个阻塞时间比请求时间要稍微长点,所以会大于5秒。

6、复杂的动作服务器

对于动作有了基本了解之后,我们就重点关注它的特点,也就是动作跟服务的区别,主要是动作的异步特性,接下来修改上述代码,实现终止目标、处理打断请求和实时反馈等功能,突出动作的优势。

cd ~/catkin_ws/src/test/src
gedit action_server2.py
#!/usr/bin/env python
# -*- coding: utf-8

import rospy
import time
import actionlib
from test.msg import TimerAction,TimerGoal,TimerResult,TimerFeedback

def do_timer(goal):
    start_time = time.time()
    update_count = 0
    
    if goal.time_to_wait.to_sec() > 60:
        result = TimerResult()
        result.time_elapsed = rospy.Duration.from_sec(time.time()-start_time)
        result.updates_sent = update_count
        server.set_aborted(result,"超时终止")
        return
    while (time.time()-start_time) < goal.time_to_wait.to_sec():
        if server.is_preempt_requested():
            result = TimerResult()
            result.time_elapsed = rospy.Duration.from_sec(time.time()-start_time)
            result.updates_sent = update_count
            server.set_preempted(result,"中断")
            return

        feedback = TimerFeedback()
        feedback.time_elapsed = rospy.Duration.from_sec(time.time()-start_time)
        feedback.time_remaining = goal.time_to_wait - feedback.time_elapsed
        server.publish_feedback(feedback)
        update_count += 1
        time.sleep(1)

    result = TimerResult()
    result.time_elapsed = rospy.Duration.from_sec(time.time()-start_time)
    result.updates_sent = update_count
    server.set_succeeded(result,"成功完成")

rospy.init_node("timer_action_server2")
server = actionlib.SimpleActionServer('timer',TimerAction,do_timer,False)
server.start()
rospy.spin()

加个执行权限:chmod u+x action_server2.py

由于需要提供反馈,所以导入TimerFeedback消息类型,对于请求time_to_wait超过60秒的,显式调用set_aborted来终止当前的目标请求,这个调用会给客户端发送一个消息,告知本次目标已被终止。如果没超时就进入到一个循环,在循环中进行间断的休眠等待,在这个过程中检查是否有发生中断is_preempt_requested等情况,并提供反馈。当休眠时长超过请求目标时长,就退出循环,告诉客户端目标已被成功执行完了。 

7、复杂的动作客户端

服务端做了调整之后,我们客户端对其进行测试,比如对反馈进行处理,中断正在执行的目标,以及引发终止等操作:

cd ~/catkin_ws/src/test/src
gedit action_client2.py
#!/usr/bin/env python
# -*- coding: utf-8

import rospy
import time
import actionlib
from test.msg import TimerAction,TimerGoal,TimerResult,TimerFeedback

def feedback_cb(feedback):
    print("持续时间:%f"%(feedback.time_elapsed.to_sec()))
    print("剩余时间:%f"%(feedback.time_remaining.to_sec()))

rospy.init_node('timer_action_client2')
client = actionlib.SimpleActionClient('timer',TimerAction)
client.wait_for_server()

goal = TimerGoal()
goal.time_to_wait = rospy.Duration.from_sec(5)
#goal.time_to_wait = rospy.Duration.from_sec(100)
client.send_goal(goal,feedback_cb=feedback_cb)

time.sleep(3)
client.cancel_goal()

client.wait_for_result()
print("状态码:%d"%(client.get_state()))
print("状态描述:%s"%(client.get_goal_status_text()))
print("持续时间:%f"%(client.get_result().time_elapsed.to_sec()))
print("反馈次数:%d"%(client.get_result().updates_sent))

添加执行权限:chmod u+x action_client2.py

查看反馈的话题的信息

rostopic info /timer/feedback

'''
Type: test/TimerActionFeedback

Publishers: 
 * /timer_action_server (http://YAB:33125/)

Subscribers: None
'''

 其消息类型是TimerActionFeedback,我们查看下详情

rosmsg show TimerActionFeedback
'''
[test/TimerActionFeedback]:
std_msgs/Header header
  uint32 seq
  time stamp
  string frame_id
actionlib_msgs/GoalStatus status
  uint8 PENDING=0
  uint8 ACTIVE=1
  uint8 PREEMPTED=2
  uint8 SUCCEEDED=3
  uint8 ABORTED=4
  uint8 REJECTED=5
  uint8 PREEMPTING=6
  uint8 RECALLING=7
  uint8 RECALLED=8
  uint8 LOST=9
  actionlib_msgs/GoalID goal_id
    time stamp
    string id
  uint8 status
  string text
test/TimerFeedback feedback
  duration time_elapsed
  duration time_remaining
'''

除了定义的两个持续时间与剩余时间之外,我们还看到了当前的可能状态有10种,主要关注其中三种:PREEMPTED=2、SUCCEEDED=3、ABORTED=4

了解了客户端的一些定义之后,我们来运行测试下(前提是先开启roscoreaction_server2.py

 

7.1、正常测试

运行客户端:rosrun test action_client2.py

持续时间:0.000023
剩余时间:4.999977
持续时间:1.001364
剩余时间:3.998636
持续时间:2.002950
剩余时间:2.997050
持续时间:3.004577
剩余时间:1.995423
持续时间:4.005102
剩余时间:0.994898
状态码:3
状态描述:成功完成
持续时间:5.007248
反馈次数:5

7.2、终止测试

将时间调到超过60秒,比如100秒

goal.time_to_wait = rospy.Duration.from_sec(100)

运行客户端:rosrun test action_client2.py

状态码:4
状态描述:超时终止
持续时间:0.000014
反馈次数:0

7.3、中断测试 

将下面代码中的注释去掉:

time.sleep(3)
client.cancel_goal()

运行客户端:rosrun test action_client2.py

持续时间:0.000042
剩余时间:4.999958
持续时间:1.002338
剩余时间:3.997662
持续时间:2.005545
剩余时间:2.994455
状态码:2
状态描述:中断
持续时间:3.007982
反馈次数:3

8、小结

节点通信的三种机制就介绍完了,其中本节说的动作机制是ROS中一个功能强大,使用广泛的通信工具。那么在哪些场景选择哪种通信,这里做一个对比:

类型场景
话题接收方有多个的情况,比如传感器的数据,单工通信。
服务请求/响应的交互式场景,计算时间短,比如询问节点的当前状态
动作大部分的请求/响应交互式场景,尤其在执行过程中不能立即完成时,比如说导航到一个目标点

这里主要是服务和动作有点暧昧,那么这么去理解,在任何时候,当你执行一个任务,想要使用服务的时候,都值得考虑是否使用动作。虽然动作需要写更多的代码,但是比服务强大很多,扩展性也好很多。 

另外代码中出现了中文,所以需要加上:# -*- coding: utf-8

在本节,我们新建了两个节点,都开启之后,我们来查看下节点列表

rosnode list
'''
/rosout
/timer_action_server
/timer_action_server2
'''

 可以看到出现了代码中定义的两个节点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/971203.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

20.添加HTTP模块

添加一个简单的静态HTTP。 这里默认读者是熟悉http协议的。 来看看http请求Request的例子 客户端发送一个HTTP请求到服务器的请求消息&#xff0c;其包括&#xff1a;请求行、请求头部、空行、请求数据。 HTTP之响应消息Response 服务器接收并处理客户端发过来的请求后会返…

尚硅谷SpringMVC (9-13)

九、HttpMessageConverter HttpMessageConverter &#xff0c;报文信息转换器&#xff0c;将请求报文转换为 Java 对象&#xff0c;或将 Java 对象转换为响应报文 HttpMessageConverter提供了两个注解和两个类型&#xff1a; RequestBody &#xff0c; ResponseBody &#xff…

如何创建美观的邮件模板并通过qq邮箱的SMTP服务向用户发送

最近在写注册功能的自动发送邮箱告知验证码的功能&#xff0c;无奈根本没有学过前端&#xff0c;只有写Qt的qss基础&#xff0c;只好借助网页设计自己想要的邮箱格式&#xff0c;最终效果如下: 也推销一下自己的项目ShaderLab&#xff0c;可运行ShaderToy上的大部分着色器代码&…

Weblogic漏洞(二)之 Weblogic漏洞环境安装

Weblogic漏洞环境安装 这里我们使用Kali虚拟机安装docker并搭建vulhub靶场来进行Weblogic漏洞环境的安装 安装 docker #更新软件 apt-get update #安装CA证书 apt-get install -y apt-transport-https ca-certificates #安装docker apt install docker.io #查看docker是否安…

【商业案例应用】C端产品设计流程——团购产品案例

文章目录 1、项目背景介绍2、产品前期分析3、产品规划4、总结 1、项目背景介绍 2、产品前期分析 3、产品规划 4、总结

SSH连接安卓手机Termux —— Android手机

文章目录 前言1.安装ssh2.安装cpolar内网穿透3.远程ssh连接配置4.公网远程连接5.固定远程连接地址 前言 使用安卓机跑东西的时候&#xff0c;屏幕太小&#xff0c;有时候操作不习惯。不过我们可以开启ssh&#xff0c;使用电脑PC端SSH远程连接手机termux。 本次教程主要实现在…

C语言入门 Day_13 二维数组

目录 前言&#xff1a; 1.字符串 2.创建二维数组 3.使用二维数组 4.易错点 5.思维导图 前言&#xff1a; 我们学习了字符类型char&#xff0c;我们可以用char来表示一个大写或者小写的字母&#xff0c;但真实应用中我们往往使用的是多个字符组成的一个单词或者句子。 …

Linux gdb单步调试的原理

文章目录 一、demo演示二、原理分析参考资料 一、demo演示 .section .data message:.string "Hello, World!\n" len . - message.section .text .globl _start _start:# 调用 write() 函数输出 "Hello, World!"mov $1, %rax # 系统调用号为 1…

海域可视化监管:浅析海域动态远程视频智能监管平台的构建方案

一、方案背景 随着科技的不断进步&#xff0c;智慧海域管理平台已经成为海洋领域监管的一种重要工具。相比传统的视频监控方式&#xff0c;智慧海域管理平台通过建设近岸海域视频监控网、海洋环境监测网和海上目标探测网络等&#xff0c;可实现海洋管理的数字化转型。 传统的…

『好书推荐』|《Effective软件测试》

作者简介 《Effective软件测试》 是一本由清华大学出版社出版的图书&#xff0c;作者是[荷]毛里西奥阿尼什&#xff08;Maurcio Aniche&#xff09;&#xff0c;译者是朱少民、李洁、张元。是2023年6月新推出的一本书籍。 Maurcio Aniche博士是荷兰代尔夫特理工大学软件工程系的…

页面页脚部分CSS分享

先看效果&#xff1a; CSS部分&#xff1a;&#xff08;查看更多&#xff09; <style>body {display: grid;grid-template-rows: 1fr 10rem auto;grid-template-areas: "main" "." "footer";overflow-x: hidden;background: #F5F7FA;min…

使用LightPicture开源搭建私人图床:详细教程及远程访问配置方法

文章目录 1.前言2. Lightpicture网站搭建2.1. Lightpicture下载和安装2.2. Lightpicture网页测试2.3.cpolar的安装和注册 3.本地网页发布3.1.Cpolar云端设置3.2.Cpolar本地设置 4.公网访问测试5.结语 1.前言 现在的手机越来越先进&#xff0c;功能也越来越多&#xff0c;而手机…

urllib库

目录 1、简介 2、请求模块 3、解析模块 1、简介 urllib是python内置的标准库&#xff0c;无需下载&#xff0c;导入即可使用。 2、请求模块 urllib包里有一个request模块 from urllib import request# 1.request模块# 1.1发送网络请求 # urlopen() : 打开url地址 resp re…

进程的概念、组成、特征

1.概念 进程是操作系统进行资源分配的最小的单位。 2.组成 进程由PCB、程序段、数据段组成。PCB是操作系统需要的&#xff0c;而程序段和数据段是用户所需要的。 PCB是一种数据结构&#xff0c;操作系统所需的进程资源都存储在PCB中&#xff0c;PCB也是进程存在的唯一标识。…

【快应用】后台运行的快应用如何自动前台打开

【关键词】 Onhide、router、后台 【问题背景】 快应用退到后台运行后&#xff0c;隔几秒钟后&#xff0c;会自动打开跳转到某个页面&#xff0c;这种情形应该如何去定位处理&#xff1f; 【问题分析】 退到后台运行&#xff0c;再自动拉起看似很诡异&#xff0c;以为是快应…

Autofac使用(3)---AOP支持

1、Nuget引入程序集 2、扩展IInterceptor public class CusotmInterceptor : IInterceptor{/// <summary>/// 切入者逻辑/// /// 使用了Intercept 方法把 要调用的Call方法给包裹起来了/// </summary>/// <param name"invocation"></param>p…

架构设计基础设施保障IaaS弹性伸缩和无服务器计算

目录 1 高可用弹性伸缩实践2 无服务器计算&#xff08;FaaS&#xff09; 1 高可用弹性伸缩实践 背景 弹性伸缩是云服务架构的重要优势&#xff0c;能够很好的解决高并发场景下的性能瓶颈&#xff0c; 同时节省运营成本。 在 IaaS 端&#xff0c;能够弹性伸缩的最实用的产品形…

AIoT+5G改变智慧城市:揭秘智慧公厕的奇妙魅力

AIoT5G的新型智慧城市应用带来了智慧公厕的全新体验。通过智能监测、高速网络、智能调控、智慧管理等技术应用&#xff0c;公厕的舒适性、便捷性和智慧化程度得到了极大提升。可以看到的是&#xff0c;智慧公厕正逐渐激活智慧城市的生活场景&#xff0c;为城市居民带来更好的生…

无需租用云服务器:使用Linux本地搭建web服务并实现内网穿透发布公网访问的详细教程

文章目录 前言1. 本地搭建web站点2. 测试局域网访问3. 公开本地web网站3.1 安装cpolar内网穿透3.2 创建http隧道&#xff0c;指向本地80端口3.3 配置后台服务 4. 配置固定二级子域名5. 测试使用固定二级子域名访问本地web站点 前言 在web项目中,部署的web站点需要被外部访问,则…

Mac电脑其他文件占用超过一大半的内存如何清理?

mac的存储空间时不时会提示内存已满&#xff0c;查看内存占用比例最大的居然是「其他文件」&#xff0c;「其他文件」是Mac无法识别的格式文件或应用插件扩展等等...如果你想要给Mac做一次彻底的磁盘空间清理&#xff0c;首当其冲可先对「其他文件」下手&#xff0c;那么我们该…