ROS2高效学习第九章 -- ros2 bag之编程实现包读取 其二

ros2 bag之编程实现包读取

  • [1 前言和资料](#1 前言和资料)
  • [2 正文](#2 正文)
    • [2.1 bag_echo 功能介绍](#2.1 bag_echo 功能介绍)
    • [2.2 bag_operator 之 bag_echo](#2.2 bag_operator 之 bag_echo)
  • [3 总结](#3 总结)

1 前言和资料

上一篇文章ROS2高效学习第九章 -- ros2 bag之编程实现包录制 其一,我们使用 cpp 编程实现了 ros2 bag 录制,支持录制任意消息类型的 topic。针对录制下来的 bag,我们可以使用 ros2 bag info ***db3 查看统计信息。但如果想离线查看 bag 里的具体 topic 数据,ros2 就没有提供相应的命令。

在更早一点的文章ROS高效入门第九章 -- 掌握rosbag工具,编写python包录制、读取、截断和回放样例中,我们使用 python 编程实现了离线查看 ros1 bag 数据功能。本文我们将复用ROS2高效学习第九章 -- ros2 bag之编程实现包录制 其一中的 bag_operator 软件包,使用 ros2 的接口,实现 ros2 bag 的离线查看。

本文参考资料如下:

(1)ROS高效入门第九章 -- 掌握rosbag工具,编写python包录制、读取、截断和回放样例 第2.2节

(2)Reading-From-A-Bag-File-CPP

(3)Recording-A-Bag-From-Your-Own-Node-Py

2 正文

2.1 bag_echo 功能介绍

(1)bag_echo 支持 -b 指定要查看的 ros2 bag;支持 -t 指定要查看的 topic,可以一次指定多个 topic,也可以不指定,默认查看所有 topic 数据。

2.2 bag_operator 之 bag_echo

(1)在 bag_operator 中创建 bag_echo.py

bash 复制代码
cd ~/colcon_ws/src/bag_operator 
mkdir scripts
touch scripts/bag_echo.py

(2)编写 bag_echo.py:由于 ros2 bag 使用了sqlite3 存储,因此接口调用要比 ros1 bag 复杂一些。

python 复制代码
#! /usr/bin/env python3
# -*- coding: utf-8 -*-

import os
import argparse
import logging
import rosbag2_py
from rclpy.serialization import deserialize_message
from rosidl_runtime_py.utilities import get_message

logging.basicConfig(level=logging.DEBUG)

class BagEcho(object):
    def __init__(self, bag, topics):
        self._bag = bag
        self._topics = topics
        self._reader = rosbag2_py.SequentialReader()

    def echo_bag(self):
        # 以sqlite3格式打开ros2 bag
        storage_options = rosbag2_py._storage.StorageOptions(uri=self._bag, storage_id='sqlite3')
        converter_options = rosbag2_py._storage.ConverterOptions('', '')
        self._reader.open(storage_options, converter_options)
        # 设置要读取的topic作为过滤条件,如果没有指定topic,则读取所有topic
        if self._topics:
            storage_filter = rosbag2_py._storage.StorageFilter(topics=self._topics)
            self._reader.set_filter(storage_filter)
        # 获取所有话题及其类型,并使用map存储
        topic_name2type = {topic_metadata.name: topic_metadata.type for topic_metadata in self._reader.get_all_topics_and_types()}

        # 循环读取消息,并进行个数统计
        msg_cnts = {}        
        while self._reader.has_next():
            (topic_name, data, t) = self._reader.read_next()

            # 统计每个话题的消息数
            if topic_name not in msg_cnts:  
                msg_cnts[topic_name] = 1
            else:
                msg_cnts[topic_name] += 1            

            # 反序列化消息
            message_type = get_message(topic_name2type[topic_name])
            msg = deserialize_message(data, message_type)

            # 打印消息内容
            print("----------[%s : %s]---------" % (topic_name, topic_name2type[topic_name]))
            print(msg)            

        # 打印摘要信息
        logging.info("----------[rosbag summary]---------")
        for topic_name in msg_cnts:  
            logging.info("topic %s msg cnt is %d" %(topic_name, msg_cnts[topic_name]))

def main():
  parser = argparse.ArgumentParser(description="echo ros2 bag topic")
  parser.add_argument("-b", "--bag", type=str, required=True, help="specify rosbag")
  parser.add_argument("-t", "--topics", nargs="+", type=str, help="specify topic")
  # extra_args可以让-t指定多个topic,并使用argparse.SUPPRESS隐藏extra_args参数
  parser.add_argument('extra_args', nargs='*', help=argparse.SUPPRESS)
  args=parser.parse_args()

  if not os.path.isfile(args.bag):
      logging.error("%s is not found!" %args.bag)
      return

  echo = BagEcho(args.bag, args.topics)
  echo.echo_bag()

if __name__ == "__main__":
    main()

(3)添加 CMakeLists.txt

c 复制代码
install(PROGRAMS
  scripts/bag_echo.py
  DESTINATION lib/${PROJECT_NAME})

(4)编译并运行

bash 复制代码
~/colcon_ws
colcon build --packages-select bag_operator
source install/local_setup.bash
# 指定多个 topic
./install/bag_operator/lib/bag_operator/bag_echo.py -b 2024-03-14-18-09-20-bag/2024-03-14-18-09-20-bag_0.db3 -t /turtle1/cmd_vel /turtle1/pose
# 不指定 topic,查看所有数据
./install/bag_operator/lib/bag_operator/bag_echo.py -b 2024-03-14-18-09-20-bag/2024-03-14-18-09-20-bag_0.db3


3 总结

本文代码托管在本人的github上:bag_operator

相关推荐
AC赳赳老秦2 分钟前
2026 智能制造趋势:DeepSeek 助力“黑灯”工厂运营,实现生产流程自动化
网络·数据结构·算法·安全·web安全·prometheus·deepseek
流云鹤6 分钟前
2026牛客寒假算法基础集训营6(K H G B A)
算法
程序员酥皮蛋7 分钟前
hot 100 第三十题 30. 两两交换链表中的节点
数据结构·算法·leetcode·链表
寻寻觅觅☆9 分钟前
东华OJ-基础题-131-8皇后·改(C++)
c++·算法·深度优先
程序员徐师兄17 分钟前
基于 Python 深度学习的电影评论情感分析算法
python·深度学习·算法·电影情感分析算法·评论情感分析
承渊政道25 分钟前
Linux系统学习【深入剖析Git的原理和使用(下)】
linux·服务器·git·学习·gitee·vim·gitcode
ShineWinsu30 分钟前
对于C++中list的详细介绍
开发语言·数据结构·c++·算法·面试·stl·list
_OP_CHEN31 分钟前
【算法提高篇】(三)线段树之维护更多的信息:从基础到进阶的灵活运用
算法·蓝桥杯·线段树·c/c++·区间查询·acm/icpc·信息维护
mjhcsp32 分钟前
C++Lyndon 分解超详解析
c++·算法·lyndon
海兰40 分钟前
elasticsearch学习之基本概念-向量数据库
数据库·学习·elasticsearch