ROS自定义话题消息

自定义话题消息

    • 定义msg文件
    • 添加功能包的依赖
    • 如何使用自定义话题消息

定义msg文件

在catkin_ws/src/中相应功能包下创建文件夹msg,该文件用于存储.msg文件,注意此文件是与编程语言无关的。/.msg文件中的内容相当于是宏定义。

byw@byw-virtual-machine:~/catkin_ws/src/learing_topic/msg$ touch Person.msg

下列代码中的string 、uint8这些表示的是需要在不同的编程语言中扩展成相应语言可以使用的数据类型,而这个过程是在执行编译时进行的。所以需要根据这是数据接口设置一些编译规则。

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

ROS自定义话题消息_第1张图片

添加功能包的依赖

在package.xml中添加功能包依赖。分别添加一个编译依赖包message_generation和一个执行依赖包message_runtime。

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

ROS自定义话题消息_第2张图片
第一,因为在package.xml中添加了一个依赖包,而在创建功能包时后面跟的依赖包不仅在package.xml中还在CMakeLists.txt中,所以还要添加到CMakeLists.txt中。
ROS自定义话题消息_第3张图片
在这里插入图片描述

第二,前面讲到需要将/.msg文件编译成对应的不同文件配置项。
add_message_files是把定义的Person.msg作为编译接口,然后编译的时候针对该文件去编译;
generate_messages是在编译Person.msg时需要哪些ros已有的包,这里需要std_msgs,上文提到的string,uint8都是这里面定义的。

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

ROS自定义话题消息_第4张图片
回到工作空间根目录进行编译,可以在编译文件夹./devel/include中查看编译产生的代码文件。注意要在编译成功后才有该c++头文件。
ROS自定义话题消息_第5张图片

如何使用自定义话题消息

C++:person_publisher.cpp

/**
 * 该例程将发布/person_info话题,自定义消息类型learning_topic::Person
 */
#include 
#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_subscriber.cpp

**
 * 该例程将订阅/person_info话题,自定义消息类型learning_topic::Person
 */
#include 

#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();

由于是.cpp文件需要编译后才可以执行,所以添加编译配置。
• 设置需要编译的代码和生成的可执行文件;
• 设置链接库(执行编译生成的可执行文件时需要的库);
• 添加依赖项(由于有些代码时动态生成的,所以可执行文件也要和刚才动态生成的头文件链接)。

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)

ROS自定义话题消息_第6张图片
在编译完成后,就可执行两个节点了。但不要忘记在rosrun之前需要roscore.

roscore

ROS自定义话题消息_第7张图片

你可能感兴趣的:(ROS,编程语言,c++,ubuntu)