ROS学习笔记(9)——话题通信自定义msg(C++篇)

一、创建msg文件

(1)在自定义功能包下,创建msg文件夹,并在msg文件夹下创建msg文件,编写文件内容,如个人信息:姓名  身高  体重

(2)保存msg文件后,打开与功能包同级的package.xml文件,修改依赖:

其中,message_generation是编译依赖,若出错则编译出现问题,message_runtime是运行依赖,若出错,则程序运行报错。

(3)打开与功能包同级的CMakeLists.text文件,从上往下修改与message有关的代码。首先添加依赖:

再修改文件名:

去掉注释:

添加依赖:

保存后,进行编译:

编译成功。

二、创建发布文件

(1)配置头文件,在工作空间devel的include文件夹下打开终端,在终端输入指令:

pwd

复制终端输出的文件路径,打开.vscode文件夹下的c_cpp_properties.json文件,将文件路径复制进“includePath”中:

可以将include/后改为“**”,意思是包含include文件夹下所有文件:

(2)编写发布程序。在功能包的src文件夹下新建一个.cpp文件,写入发布程序:

#include "ros/ros.h"
#include "plumbing_pub_sub/person.h"
/*
    发布方:发布人的消息
        1.包含头文件
            #include "plumbing_pub_sub/person.h"
        2.初始化ROS节点
        3.创建ROS节点句柄
        4.创建发布者对象
        5.编写发布逻辑,发布数据
*/
int main(int argc, char *argv[])
{
    setlocale(LC_ALL,"");
    //2.初始化ROS节点
    ros::init(argc,argv,"Laoshi");
    //3.创建ROS节点句柄
    ros::NodeHandle nh;
    //4.创建发布者对象
    ros::Publisher pub = nh.advertise<plumbing_pub_sub::person>("liaotian",10);
    //5.编写发布逻辑,发布数据
    //(1)创建被发布的数据
    plumbing_pub_sub::person person;
    person.name = "张三";
    person.age = 0;
    person.height = 1.68;
    //(2)设置发布频率
    ros::Rate rate(1);
    //(3)循环发布数据
    while(ros::ok()){
        //修改数据
        person.age += 1;
        //核心:数据发布
        pub.publish(person);
        ROS_INFO("发布的消息:%s,%d,%.2f",person.name.c_str(),person.age,person.height);
        //休眠
        rate.sleep();
        //建议
        ros::spinOnce();
    }
    return 0;
}

(3)开与功能包同级的CMakeLists.text文件,修改文件名:

添加文件依赖:

(4)保存后编译:

(5)打开终端,先执行roscore,配置环境后执行文件:

(6)为了更加直观,可以将输出结果打印在屏幕上,在数据发布处添加一行代码:

ROS_INFO("发布的消息:%s,%d,%.2f",person.name.c_str(),person.age,person.height);

再次运行:

运行成功,可以看见输出结果打印在屏幕上,无问题。

三、创建订阅文件

(1)在功能包src文件夹下创建.cpp文件命名为demo04_sub_person.cpp,编写订阅文件代码:

#include "ros/ros.h"
#include "plumbing_pub_sub/person.h"
/*
    订阅方:订阅者的消息
        1.包含头文件
            #include "plumbing_pub_sub/person.h"
        2.初始化ROS节点
        3.创建ROS节点句柄
        4.创建订阅者对象
        5.处理订阅的数据
        6.调用spin()函数
*/
void doPerson(const plumbing_pub_sub::person::ConstPtr& person){
    ROS_INFO("订阅人的信息:%s,%d,%.2f",person->name.c_str(),person->age,person->height);
}
int main(int argc, char *argv[])
{
    setlocale(LC_ALL,"");
    ROS_INFO("订阅方实现:");
    // 2.初始化ROS节点
    ros::init(argc,argv,"jiazhang");
    // 3.创建ROS节点句柄
    ros::NodeHandle nh;
    // 4.创建订阅者对象
    ros::Subscriber sub = nh.subscribe("liaotian",10,doPerson);
    // 5.处理订阅的数据
    // 6.调用spin()函数
    ros::spin();




    return 0;
}

(2)开与功能包同级的CMakeLists.text文件,修改文件名:

(3)保存后编译,编译无问题后,打开终端,运行roscore,配置环境,运行订阅程序和发布程序:

运行成功。

### 回答1: Protobuf是一种轻量级的数据交换格式,ROS是一个机器人操作系统,它提供了强大的机器人开发和部署平台。为了在ROS系统中使用protobuf数据格式,需要将protobuf转换为ROS msg数据格式。以下是如何进行protobuf转ROS 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: protobuf和ros 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的各种工具进行开发,大大简化了ROS与protobuf之间的通信。 ### 回答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消息更加冗长,但在某些情况下,这种转换可能是必需的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值