创建ROS消息和ROS服务(九)

本教程详细介绍如何创建并编译ROS消息和服务,以及rosmsg, rossrv和roscp命令行工具的使用。

内容

    消息(msg)和服务(srv)介绍
使用 msg
    创建一个 msg
    使用 rosmsg
使用 srv
    创建一个srv
    使用 rossrv
msg和srv都需要的步骤
获得帮助
回顾
下一个教程

消息(msg)和服务(srv)介绍

消息(msg): msg文件就是一个描述ROS中所使用消息类型的简单文本。它们会被用来生成不同语言的源代码。

服务(srv): 一个srv文件描述一项服务。它包含两个部分:请求和响应。 

msg文件存放在package的msg目录下,srv文件则存放在srv目录下。

msg文件实际上就是每行声明一个数据类型和变量名。可以使用的数据类型如下:

int8, int16, int32, int64 (plus uint*)
float32, float64
string
time, duration
other msg files
variable-length array[] and fixed-length array[C] 

在ROS中有一个特殊的数据类型:Header,它含有时间戳和坐标系信息。在msg文件的第一行经常可以看到Header header的声明.

下面是一个msg文件的样例,它使用了Header,string,和其他另外两个消息类型。

  Header header
  string child_frame_id
  geometry_msgs/PoseWithCovariance pose
  geometry_msgs/TwistWithCovariance twist

srv文件分为请求和响应两部分,由’—’分隔。下面是srv的一个样例:

int64 A
int64 B
——
int64 Sum

其中 A 和 B 是请求, 而Sum 是响应。

使用 msg

创建一个 msg

下面,我们将在之前创建的package里定义新的消息。

$ cd ~/catkin_ws/src/beginner_tutorials
$ mkdir msg
$ echo "int64 num" > msg/Num.msg

上面是最简单的例子——在.msg文件中只有一行数据。当然,你可以仿造上面的形式多增加几行以得到更为复杂的消息:

string first_name
string last_name
uint8 age
uint32 score

接下来,还有关键的一步:我们要确保msg文件被转换成为C++,Python和其他语言的源代码:

查看package.xml, 确保它包含一下两条语句:

message_generation
message_runtime

如果没有,添加进去。 注意,在构建的时候,我们只需要”message_generation”。然而,在运行的时候,我们只需要”message_runtime”。

在你最喜爱的编辑器中打开CMakeLists.txt文件(可以参考前边的教程rosed).

在 CMakeLists.txt文件中,利用find_packag函数,增加对message_generation的依赖,这样就可以生成消息了。 你可以直接在COMPONENTS的列表里增加message_generation,就像这样:

# Do not just add this line to your CMakeLists.txt, modify the existing line
find_package(catkin REQUIRED COMPONENTS roscpp rospy std_msgs message_generation)

有时候你会发现,即使你没有调用find_package,你也可以编译通过。这是因为catkin把你所有的package都整合在一起,因此,如果其他的package调用了find_package,你的package的依赖就会是同样的配置。但是,在你单独编译时,忘记调用find_package会很容易出错。

同样,你需要确保你设置了运行依赖:

catkin_package(
  ...
  CATKIN_DEPENDS message_runtime ...
  ...)

找到如下代码块:

# add_message_files(
#   FILES
#   Message1.msg
#   Message2.msg
# )

去掉注释符号#,用你的.msg文件替代Message*.msg,就像下边这样:

add_message_files(
  FILES
  Num.msg
)

手动添加.msg文件后,我们要确保CMake知道在什么时候重新配置我们的project。 确保添加了如下代码:

generate_messages()

现在,你可以生成自己的消息源代码了。如果你想立即实现,那么就跳过以下部分,到Common step for msg and srv.

使用 rosmsg

以上就是你创建消息的所有步骤。下面通过rosmsg show命令,检查ROS是否能够识消息。

使用方法:

$ rosmsg show [message type]

样例:

$ rosmsg show beginner_tutorials/Num

你将会看到:

    int64 num

在上边的样例中,消息类型包含两部分:

beginner_tutorials -- 消息所在的package

Num -- 消息名Num. 

如果你忘记了消息所在的package,你也可以省略掉package名。输入:

$ rosmsg show Num

你将会看到:

[beginner_tutorials/Num]:
int64 num

使用 srv

创建一个srv

在刚刚那个package中创建一个服务:

$ roscd beginner_tutorials
$ mkdir srv

这次我们不再手动创建服务,而是从其他的package中复制一个服务。 roscp是一个很实用的命令行工具,它实现了将文件从一个package复制到另外一个package的功能。

使用方法:

$ roscp [package_name] [file_to_copy_path] [copy_path]

现在我们可以从rospy_tutorials package中复制一个服务文件了:

$ roscp rospy_tutorials AddTwoInts.srv srv/AddTwoInts.srv

还有很关键的一步:我们要确保srv文件被转换成C++,Python和其他语言的源代码。

现在认为,你已经如前边所介绍的,在CMakeLists.txt文件中增加了对message_generation的依赖。:

# Do not just add this line to your CMakeLists.txt, modify the existing line
find_package(catkin REQUIRED COMPONENTS roscpp rospy std_msgs message_generation)

(对的, message_generation 对msg和srv都起作用)

同样,跟msg文件类似,你也需要在package.xml文件中做一些修改。查看上边的说明,增加额外的依赖项。

删掉#,去除对下边语句的注释:

# add_service_files(
#   FILES
#   Service1.srv
#   Service2.srv
# )

用你自己的srv文件名替换掉那些Service*.srv文件:

add_service_files(
  FILES
  AddTwoInts.srv
)

现在,你可以生成自己的服务源代码了。如果你想立即实现,那么就跳过以下部分,到Common step for msg and srv.

使用 rossrv

以上就是创建一个服务所需的所有步骤。下面通过rosmsg show命令,检查ROS是否能够识该服务。

使用方法:

$ rossrv show <service type>

例子:

$ rossrv show beginner_tutorials/AddTwoInts

你将会看到:

 int64 a
    int64 b
    ---
    int64 sum

跟rosmsg类似, 你也可以不指定具体的package名来查找服务文件:

$ rossrv show AddTwoInts
[beginner_tutorials/AddTwoInts]:
int64 a
int64 b
---
int64 sum

[rospy_tutorials/AddTwoInts]:
int64 a
int64 b
---
int64 sum

msg和srv都需要的步骤

接下来,在CMakeLists.txt中找到如下部分:

# generate_messages(
#   DEPENDENCIES
# #  std_msgs  # Or other packages containing msgs
# )

去掉注释并附加上所有你消息文件所依赖的那些含有.msg文件的package(这个例子是依赖std_msgs,不要添加roscpp,rospy),结果如下:

generate_messages(
  DEPENDENCIES
  std_msgs
)

由于增加了新的消息,所以我们需要重新编译我们的package:

# In your catkin workspace
$ cd ../..
$ catkin_make
$ cd -

所有在msg路径下的.msg文件都将转换为ROS所支持语言的源代码。生成的C++头文件将会放置在~/catkin_ws/devel/include/beginner_tutorials/。 Python脚本语言会在 ~/catkin_ws/devel/lib/python2.7/dist-packages/beginner_tutorials/msg 目录下创建。 lisp文件会出现在 ~/catkin_ws/devel/share/common-lisp/ros/beginner_tutorials/msg/ 路径下.

详尽的消息格式请参考Message Description Language 页面.

获得帮助

我们已经接触到不少的ROS工具了。有时候很难记住他们所需要的参数。还好大多数ROS工具都提供了帮助。

输入:

$ rosmsg -h
你可以看到一系列的rosmsg子命令.

Commands:
  rosmsg show Show message description
  rosmsg users  Find files that use message
  rosmsg md5  Display message md5sum
  rosmsg package  List messages in a package
  rosmsg packages List packages that contain messages

同样你也可以获得子命令的帮助:

$ rosmsg show -h
这会现实rosmsg show 所需的参数:

Usage: rosmsg show [options] <message type>

Options:
  -h, --help  show this help message and exit
  -r, --raw   show raw message text, including comments

回顾

总结一下到目前为止我们接触过的一些命令:

    rospack = ros+pack(age) : provides information related to ROS packages
    rosstack = ros+stack : provides information related to ROS stacks

    roscd = ros+cd : changes directory to a ROS package or stack

    rosls = ros+ls : lists files in a ROS package

    roscp = ros+cp : copies files from/to a ROS package
    rosmsg = ros+msg : provides information related to ROS message definitions
    rossrv = ros+srv : provides information related to ROS service definitions
    rosmake = ros+make : makes (compiles) a ROS package 
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ROS创建服务server和client的步骤如下: 1. 创建服务文件:在ROS包中创建一个srv文件,定义服务的请求和响应类型。 2. 编写服务server:在ROS节点中编写服务server的代码,实现服务的功能。 3. 编写服务client:在ROS节点中编写服务client的代码,调用服务server提供的功能。 4. 编译ROS包:使用catkin_make命令编译ROS包,生成可执行文件。 5. 运行ROS节点:使用rosrun命令运行ROS节点,启动服务server和client。 6. 测试服务:使用rostopic命令发布服务请求,查看服务响应结果。 以上是ROS创建服务server和client的基本步骤,具体实现可以参考ROS官方文档和相关教程。 ### 回答2: 首先,ROS是一个流行的机器人操作系统,具有强大的工具集来创建和管理机器人应用程序。其中,服务(Service)是ROS中实现通信的另一个重要机制。services允许节点之间进行同步通信,一个节点可以请求一个service,另一个节点会响应请求。在ROS中,service分别有server端和client端两个组成部分,下面我们来介绍一下ROS创建service的具体步骤: 1. 编写srv文件 在ROS中,服务是由srv文件定义的,类似于msg文件。srv文件定义服务的请求和响应格式。我们可以通过文本编辑器在ROS工作空间中创建srv文件,例如在package中创建test_srv.srv文件: ``` int64 A int64 B --- int64 C ``` 其中,`A`、`B`和`C`均为服务变量的名称,`int64`代表对应变量类型。这里定义了A和B两个变量作为服务请求的输入,C变量作为服务的响应结果。 2. 编写server端代码 在ROS中,一个节点往往同时充当server和client。因此在编写server端代码之前,需要创建一个node。可以使用`rospy.init_node()`初始化一个节点。然后,在Python中,可以使用`rospy.Service()`函数来定义server服务和回调函数: ``` #!/usr/bin/env python import rospy from test_srv.srv import TestSrvResponse, TestSrvRequest def srv_callback(req): res = TestSrvResponse() res.out = req.A + req.B return res if __name__ == '__main__': rospy.init_node('test_server') srv = rospy.Service('test_service', TestSrv, srv_callback) rospy.spin() ``` 上面代码中,`srv_callback()`函数就是用于处理服务请求的服务回调函数,用户需要根据自身服务逻辑来编写。再将`srv_callback()`绑定到`rospy.Service()`中完成服务的定义和注册。 `rospy.spin()`函数保证了服务一直处于等待请求的状态直到stop节点。 3. 编写client端代码 在ROS中,client可以使用C++或Python编写,这里以Python为例。首先需要导入所用的srv定义。 在Python中使用服务,可以利用rospy.ServiceProxy()函数来创建一个service的proxy,并且传递必要的参数来调用该service: ``` #!/usr/bin/env python import rospy from test_srv.srv import TestSrvResponse, TestSrvRequest if __name__ == '__main__': rospy.init_node('test_client') rospy.wait_for_service('test_service') test_srv = rospy.ServiceProxy('test_service', TestSrv) req = TestSrvRequest() req.A = 3 req.B = 4 res = test_srv(req) rospy.loginfo("server response:{0}".format(res.out)) ``` 在Python脚本中,使用`rospy.wait_for_service()`函数等待一个服务,然后,创建TestSrvRequest类型的请求,发送给已定义的服务,并等待服务响应。 最后,本文仅给出了ROS服务server和client的关键部分代码,完整代码详见ROS官网。服务的安全性和错误处理也是极重要的,使用时需要谨慎,精心编写容错处理,以保证服务相互安全。 ### 回答3: ROS是一个流行的机器人操作系统,它提供了构建机器人应用程序所需的工具和库。ROS中的服务提供了一种在不同节点之间共享数据的方法。服务以client-server模式工作,客户端节点可以向服务节点请求数据,服务节点会响应这些请求。这篇文章将介绍如何在ROS创建服务server和client。 首先,在ROS创建服务必须要在package中创建srv文件,这些srv文件定义了服务消息的类型。服务消息定义了服务节点可以接受和返回的数据类型。在srv文件中,必须指定请求和响应数据类型的详细信息。在本文中,我们将创建一个服务节点来处理两个整数之间的加法操作。 例如,在我们的my_service_pkg包中创建一个名称为add_two_ints.srv的srv文件。该srv文件的格式如下所示: int64 a int64 b --- int64 sum 在上面的srv文件中,我们定义了两个整数"a"和"b"作为请求消息的参数,并定义了一个名为"sum"的相应参数。 创建完srv文件后,我们需要在CMakeLists.txt文件中添加相关内容以进行编译和构建。我们需要将"add_two_ints.srv"添加到package.xml中,以便在构建期间自动生成相关文件。 接下来,我们需要在服务节点中编写代码。在服务节点中,我们需要调用ros::ServiceServer服务类以添加服务。该类允许我们注册一个回调函数,当请求消息到达时执行该回调函数。 在ROS中,服务节点和客户端节点通常运行于不同的进程中。因此,我们需要使用rosrun命令启动两个不同的节点。一个节点将作为服务节点,另一个节点将作为客户端节点。 在客户端节点中,我们需要在代码中调用serviceClient类,以便连接到服务节点。serviceClient类允许我们使用类似函数调用的语法从服务节点请求数据。 在代码中使用ROS服务client和server非常简单,下面是一个基本的示例: **服务server** ``` #include "ros/ros.h" #include "my_service_pkg/addTwoInts.h" bool add(my_service_pkg::addTwoInts::Request& req, my_service_pkg::addTwoInts::Response& res) { 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::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; } ``` 在上述示例中,我们创建了一个名为"add"的回调函数,当请求到达时执行该函数。该回调函数从请求参数中读取两个整数"a"和"b",并将它们相加,然后将结果存储在响应参数中。最后,该函数将响应发送回客户端节点。 **服务client** ``` #include "ros/ros.h" #include "my_service_pkg/addTwoInts.h" int main(int argc, char **argv) { ros::init(argc, argv, "add_two_ints_client"); ros::NodeHandle n; ros::ServiceClient client = n.serviceClient<my_service_pkg::addTwoInts>("add_two_ints"); my_service_pkg::addTwoInts srv; srv.request.a = 100; srv.request.b = 500; 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; } ``` 在上述示例中,我们创建了一个服务client,连接到服务节点上的"add_two_ints"服务。我们填充了一个my_service_pkg::addTwoInts类型的请求参数,并使用call函数调用服务。如果服务成功响应,则我们将打印出响应结果。 总结:在ROS中,创建服务server和client非常简单。我们定义一个srv文件,编写服务节点代码和客户端节点代码,然后启动两个不同节点,客户端节点调用服务节点并获得服务响应。服务ROS中非常有用的机制之一,可以帮助组织和解决各种机器人应用程序中的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值