ROS--话题模型(自定义消息)

本文档介绍了如何在ROS中自定义话题消息,包括创建.msg文件,配置package.xml和CMakeLists.txt,以及编写发布者和订阅者的代码。通过示例展示了创建Person消息类型,并实现发布者person_publisher和订阅者person_subscriber的完整过程。编译并运行后,两个节点能够成功通信。


参考古月老师ROS入门21讲,自己的一点笔记。

话题模型

在这里插入图片描述
在开发中,ROS定义好的消息无法满足自己的需求,需要自己定义消息。以Person信息举例,整个通讯在Master下进行。

自定义话题消息

1.在功能包目录下创建msg文件夹,内部创建.msg文件。.msg文件内容和语言无关。例如Person.msg内

string name
uint8 sex
uint8 age
uint8 unknown = 0
uint8 male = 1
uint8 female = 2

2.在功能包中的package.xml中添加依赖
编译依赖:依赖动态产生message的ROS功能包,
运行依赖:依赖动态运行时的runtime

<build_depend>message_generation</build_depend>
<exec_depend>message_runtime</exec_depend>

3.CMakeLists.txt中添加编译选项
因为在刚才添加了依赖的功能包,所以在cmake中的find package中也要添加。

find_package( message_generation)

添加把.msg编译成不同文件的配置项。第一句是把.msg作为消息接口做编译,第二句是编译这个文件需要哪些ROS已有的库文件。

add_message_files(FILES Person.msg)
generate_messages(DEPENDENCIES std_msgs)

创建在catkin_package下的运行依赖

catkin_package( message_runtime)

4.编译生成语言相关文件
在catkin_ws下编译成功后会在/devel/include/<功能包> 中根据.msg编译生成的.h文件。

创建Publisher&Subscriber的代码

person_publisher.cpp

#include <ros/ros.h>
#include "learning_topic/Person.h"
int main(int argc, char *argv[])
{
    // ROS节点初始化
    ros::init(argc, argv, "person_publisher");
    // 创建节点句柄
    ros::NodeHandle n;
    // 创建一个Publisher,发布名为/person_info的topic,消息类型为learning_topic::Person,队列长度10
    ros::Publisher person_info_pub = n.advertise<learning_topic::Person>("/person_info", 10);
    // 设置循环的频率
    ros::Rate loop_rate(1);
    int count = 0;
    while (ros::ok())
    {
        // 初始化learning_topic::Person类型的消息
    	learning_topic::Person person_msg;
		person_msg.name = "Tom";
		person_msg.age  = 18;
		person_msg.sex  = learning_topic::Person::male;
        // 发布消息
		person_info_pub.publish(person_msg);

       	ROS_INFO("Publish Person Info: name:%s  age:%d  sex:%d", 
				  person_msg.name.c_str(), person_msg.age, person_msg.sex);
        // 按照循环频率延时
        loop_rate.sleep();
    }
    return 0;
}

person_subscriber.cpp

#include <ros/ros.h>
#include "learning_topic/Person.h"
// 接收到订阅的消息后,会进入消息回调函数
void personInfoCallback(const learning_topic::Person::ConstPtr& msg)
{
    // 将接收到的消息打印出来
    ROS_INFO("Subcribe Person Info: name:%s  age:%d  sex:%d", 
			 msg->name.c_str(), msg->age, msg->sex);
}
int main(int argc, char *argv[])
{
    // 初始化ROS节点
    ros::init(argc, argv, "person_subscriber");
    // 创建节点句柄
    ros::NodeHandle n;
    // 创建一个Subscriber,订阅名为/person_info的topic,注册回调函数personInfoCallback
    ros::Subscriber person_info_sub = n.subscribe("/person_info", 10, personInfoCallback);
    // 循环等待回调函数
    ros::spin();
    return 0;
}

配置代码编译规则

配置CMakeList.txt中的编译规则
1.设置需要编译的代码和生成的可执行文件
2.设置链接库
3.添加依赖项: 因为有一些代码是动态生成的,所以必须让可执行文件和动态生成的程序产生依赖关系。

add_executable(person_publisher src/person_publisher.cpp)
target_link_libraries(person_publisher ${catkin_LIBRARIES})
add_dependencies(person_publisher ${PROJECT_NAME}_generate_messages_cpp)

add_executable(person_subscriber src/person_subscriber.cpp)
target_link_libraries(person_subscriber ${catkin_LIBRARIES})
add_dependencies(person_subscriber ${PROJECT_NAME}_generate_messages_cpp)

编译并运行

$ cd~/catkin_ws
$ catkin_make
$ source devel/setup.bash
$ roscore
$ rosrun learning_topic person_subscriber
$ rosrun learning_topic person_publisher

这时候会看到发布者和订阅者通信。
ps: ROS Master 帮助 publisher&subscriber 建立链接后,这时关闭roscore,可以发现发布者和接收者继续保持通信,但是访问参数,也无法帮助第三个节点建立连接。

ROS Serial协议是一种用于在嵌入式设备(如Arduino、STM32等)运行ROS的主机之间进行串行通信的协议。它通过标准串口(如UART、USB转串口)实现数据的双向传输,使嵌入式系统能够无缝接入ROS生态系统。该协议广泛用于机器人传感器数据采集、执行器控制、传感器驱动开发等场景。 ### ROS Serial协议的核心机制 ROS Serial协议基于串行通信接口(如RS232、TTL UART)进行数据传输,使用自定义的帧格式进行数据封装。通信双方通过预定义的协议结构进行握手、数据交换和错误校验。其通信模型包括两个主要部分:主机端(通常为运行ROS的PC)和设备端(如Arduino)。 通信过程中,主机通过串口发送请求帧,设备端响应并发送数据帧。数据帧中包含话题名称、消息类型、序列号和有效载荷等内容,确保数据在ROS系统中正确路由和解析。 ROS Serial协议采用CRC16校验机制,确保数据传输的完整性可靠性。同时,它支持多种消息类型,包括标准ROS消息(如`std_msgs`、`sensor_msgs`)以及自定义消息类型,从而实现灵活的消息交互[^1]。 ### ROS Serial的主要组件 - **rosserial_client**:提供用于嵌入式平台的客户端库,支持多种微控制器架构。 - **rosserial_server**:运行在主机端,负责设备端建立串口连接并桥接ROS系统。 - **rosserial_python**:使用Python实现的串口服务器,适用于快速原型开发。 - **rosserial_arduino**:专为Arduino平台设计的库,简化了ROS的集成过程。 ### 配置使用指南 #### 1. 安装依赖项 在Ubuntu系统上安装ROS Serial相关包: ```bash sudo apt install ros-humble-rosserial ``` 此外,如果使用Arduino作为设备端,需安装Arduino IDE并配置ROS Serial库: ```bash rosrun rosserial_arduino make_libraries.py . ``` 将生成的`ros_lib`文件夹复制到Arduino IDE的库目录中。 #### 2. 编写嵌入式端代码(以Arduino为例) 以下是一个使用`rosserial_arduino`发布的简单示例,发布一个整型消息到`chatter`话题: ```cpp #include <ros.h> #include <std_msgs/Int16.h> ros::NodeHandle nh; std_msgs::Int16 msg; ros::Publisher pub("chatter", &msg); void setup() { nh.initNode(); nh.advertise(pub); } void loop() { static int16_t i = 0; msg.data = i++; pub.publish(&msg); nh.spinOnce(); delay(1000); } ``` #### 3. 启动ROS Serial桥接节点 在主机端使用以下命令启动串口桥接节点: ```bash rosrun rosserial_python serial_node.py _port:=/dev/ttyUSB0 _baud:=57600 ``` 其中`_port`指定串口设备路径,`_baud`设置波特率。 #### 4. 查看发布的话题 启动后,可以使用以下命令查看设备端发布的话题: ```bash rostopic list rostopic echo /chatter ``` ### 通信协议帧结构 ROS Serial使用帧结构进行数据传输,每帧由以下部分组成: - **Header**:帧起始标志(0xFF) - **Length**:数据长度(不包括Header和Checksum) - **Message Type**:消息类型标识符 - **Topic ID**:话题ID,用于映射到ROS话题 - **Payload**:实际数据内容(如ROS消息序列化数据) - **Checksum**:CRC16校验值,用于数据完整性校验 ### 常见问题调试建议 - **波特率不匹配**:确保主机端设备端的波特率一致,否则会导致通信失败。 - **消息类型不匹配**:设备端发送的消息类型必须主机端订阅的消息类型一致。 - **CRC校验失败**:检查串口连接是否稳定,避免噪声干扰导致数据错误。 - **话题未正确注册**:确保在设备端调用`nh.advertise()`或`nh.subscribe()`进行话题注册。 ### 工程实践建议 - 在设计硬件时,预留串口调试接口,便于后续调试固件更新。 - 使用逻辑分析仪或串口调试工具(如`minicom`、`screen`)监控通信过程。 - 对于多设备接入系统,建议为每个设备分配独立的话题命名空间,避免命名冲突。 - 在嵌入式端使用`#define`或配置参数统一管理波特率、话题名称等关键参数。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值