在 ROS 中使用 Protobuf 替代 ros msg

本文介绍了一种在ROS环境中使用protobuf替代rosmsg的方法,解决了rosmsg扩展性差的问题,并保持了与ROS系统的高度兼容性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转自:https://segmentfault.com/a/1190000012734275

Background

做 ROS 相关开发的,应该都知道 ros msg 有个非常大的槽点:

ros msg 扩展性较差,即如果 msg 的字段发生变化,则程序前后版本不兼容

因此,google 的 protobuf 相对就是一个更好的选择。在拥有更好的扩展性的同时,还能给对数据进行压缩,减少 rosbag 的体积。

然而,ROS 的 topic 要求必须使用标准的 ros message,并不是很方便换成 protobuf。如果将 protobuf 转成 str 放在特定的 ros msg 中的话,则又失去了 msg 的类型信息。

在各种尝试下,我找了一种更好的方法,将 msg 替换成 protobuf,同时与 ROS 尽可能的兼容。

注意:本文只探讨 python 环境下的实现,c++ 版的实现由于是其他同事做的,因此不在本文的探讨范围内。

Solution

1. 通过 catkin_make 自动生成 protobuf 对应的代码

可以利用 catkin 的 add_custom_command, 来自动生成代码。这样会自动 install 到相应的目录。使用时,只要 source setup.bash 即可,完全符合 ROS 的推荐使用方式。

# 只节选最核心的部分
set(proto_dir ${PROJECT_SOURCE_DIR})
file(GLOB proto_files "${proto_dir}/*.proto")
message(STATUS "Proto Source Dir: ${proto_dir}")
message(STATUS "Proto Source Files: ${proto_files}")

catkin_destinations()
# 设置生成目标代码文件的路径
set(proto_gen_py_dir ${CATKIN_DEVEL_PREFIX}/${CATKIN_PACKAGE_PYTHON_DESTINATION})
file(MAKE_DIRECTORY ${proto_gen_py_dir})
# 这步很重要,让目标路径变为 python 的 package
# 否则的话,会出现 import 异常
file(WRITE ${proto_gen_py_dir}/__init__.py)

# Create lists of files to be generated.
set(proto_gen_py_files "")
foreach(proto_file ${proto_files})
    get_filename_component(proto_name ${proto_file} NAME_WE)
    list(APPEND proto_gen_py_files ${proto_gen_py_dir}/${proto_name}_pb2.py)
endforeach(proto_file ${proto_files})
message(STATUS "Generated proto files: ${proto_gen_py_files}")

# Run protoc and generate language-specific headers.
add_custom_command(
    OUTPUT  ${proto_gen_py_files}
    COMMAND ${PROTOBUF_PROTOC_EXECUTABLE} --proto_path=${proto_dir} --python_out=${proto_gen_py_dir}  ${proto_files}
    DEPENDS ${PROTOBUF_PROTOC_EXECUTABLE} ${proto_files}
    WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
)
set_source_files_properties(${proto_gen_py_files} PROPERTIES GENERATED TRUE)

# 将生成的 py 文件拷贝到 install 对应的路径下
install(DIRECTORY ${proto_gen_py_dir}/
  DESTINATION ${CATKIN_PACKAGE_PYTHON_DESTINATION}
  FILES_MATCHING PATTERN "*.py"
)

2. 实现一个 Adapter 类,来实现标准的 ROS message

随便找一个简单的 ros msg 所生成的 python 对应的代码,会发现其实主要做了以下几件事:

  1. 继承了 genpy.Message。在使用时会强制校验是否是 genpy.Message 的子类
  2. 实现了 serialize 方法,对内容进行序列化
  3. 实现了 deserialize 方法,对内容进行反序列化

因此,我们只要按照这种标准的方式,将 protobuf 格式的消息进行序列化或反序列化即可

### 回答1Protobuf是一种轻量级的数据交换格式,ROS是一个机器人操作系统,它提供了强大的机器人开发和部署平台。为了在ROS系统中使用protobuf数据格式,需要将protobuf转换为ROS msg数据格式。以下是如何进行protobufROS msg的过程: 首先需要安装Ros Message Generation Framework,可以使用以下命令进行安装: sudo apt-get install ros-<distro>-message-generation 其中<distro>代表ROS发行版名称,例如ROS Kinetic的<distro>值为kinetic。 接着,按照ROS消息生成的方法,定义Protobuf的消息结构并生成Protobuf消息的C++代码。在生成的代码中,可以找到MsgType.msg文件。该文件定义了ROS消息的结构。例如,对于Guigui.proto文件,将生成Guigui.pb.cc和Guigui.proto.h两个文件。在Guigui.proto.h中,找到对应MsgType.msg文件的结构。 将MsgType.msg文件复制到ROS工作空间中的msg文件夹中。例如,将Guigui.msg文件复制到catkin_ws/src/guigui/msg中。 使用catkin_make命令编译ROS工作空间,使之能够识别MsgType.msg文件。如果ROS提示找不到MsgType.msg文件,则需要重新编译工作空间。 在C++节点中调用ROS消息结构MsgType,使用ROS API函数进行消息发布和订阅。 以上是将Protobuf转换为ROS msg的过程。通过这种转换,可以在ROS系统中方便地使用Protobuf格式的数据。这对于机器人开发等需要高效数据交换的场景非常有用。 ### 回答2: protobufros msg都是常用的数据序列化协议,protobuf是Google开发的高效的二进制数据序列化库,而ROS(msg)是机器人操作系统中定义数据格式的工具,用于定义ROS节点之间传输消息的结构和内容。 将protobuf转化为ROS(msg)是一项常见的任务,在机器人应用中经常需要使用。通常,将protobuf转化为ROS(msg)的过程可以分为以下步骤: 1. 定义ROS消息类型 在ROS中,消息类型是由.msg文件定义的。需要根据protobuf中的数据结构,手动定义与之对应的.msg文件。这些文件需要指定消息的名称、数据类型以及消息字段等,这些信息决定了每个ROS消息的结构和内容。 在定义ROS消息类型时,需要确保消息名称和protobuf的名称一致,同时在定义字段时也需要按照protobuf中的数据类型定义每个字段的类型、名称、数组长度等信息。 2. 生成ROS(msg)代码 Msg文件定义完成后,需要生成ROS(msg)代码。可以使用ROS的catkin工具,编译.msg文件,生成对应的消息代码。使用的命令是"catkin_make",执行命令后,rosmsg就可以检测到新的消息类型。 3. 导入protobuf数据 将protobuf数据导入到ROS系统中,需要定义一个ROS节点,使用ROS的消息发布器来发送protobuf消息。需要在节点的代码中加载protobuf所对应的.proto文件和msg数据类型定义,并将消息发送到指定的主题。 在发送消息之前,需要将protobuf数据转换为ROS(msg)数据格式,并将其发布到ROS系统中,其他节点就能够接收到这些消息。 总之,将protobuf转化为ROS(msg)需要进行消息类型定义、生成ROS(msg)代码和消息发布等相关工作,只有完成这些步骤才可以成功。可以结合ROS的各种工具进行开发,大大简化了ROSprotobuf之间的通信。 ### 回答3: Protocol Buffers(protobuf)是一种用于序列化结构化数据以进行通信的高效数据格式。在ROS(机器人操作系统)中,ROS消息是用于在机器人系统中传输数据的标准化消息格式。在某些情况下,需要将protobuf消息转换为ROS消息。 要将protobuf消息转换为ROS消息,需要完成以下步骤: 步骤1:定义消息 首先,必须定义一个ROS消息,并确定它的数据类型、名称和字段。 例如,假设需要将具有以下protobuf定义的消息转换为ROS消息: ```proto message Person { required string name = 1; required int32 age = 2; optional string email = 3; } ``` 可以定义一个类似于以下的ROS消息来表示这一消息: ```msg string name int32 age string email ``` 步骤2:编写转换程序 接下来,需要编写一个转换程序,将protobuf格式的消息转换为ROS消息。 ```python import rospy from your_package.msg import PersonMessage # PersonMessage 为自己定义的含有name, age, email字段的 ROS message from your_package.Person_pb2 import Person # Person_pb2 为自己定义的 protobuf message def convert_to_ros(person_protobuf): person_ros = PersonMessage() person_ros.name = person_protobuf.name person_ros.age = person_protobuf.age person_ros.email = person_protobuf.email return person_ros def callback(person_protobuf): person_ros = convert_to_ros(person_protobuf) # do something with the ROS message rospy.init_node('protobuf_to_ros', anonymous=True) rospy.Subscriber('protobuf_topic', Person, callback, queue_size=1) ``` 步骤3:运行节点并测试 最后,将节点运行起来,并测试ROS消息是否成功地从protobuf消息进行了转换。 ```bash $ rosrun your_package protobuf_to_ros.py ``` 注意,对于每个protobuf消息,必须编写转换程序进行手动转换。这样的转换可能比直接使用ROS消息更加冗长,但在某些情况下,这种转换可能是必需的。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值