ros发布和订阅msg,自定义msg

文章详细介绍了如何在ROS中发布和订阅标准消息以及自定义消息。首先展示了如何创建发布者,设置发布频率,并打印发布的std_msgs::String类型消息。接着,演示了创建订阅者和回调函数来处理接收到的消息。此外,还解释了如何发布和订阅自定义消息类型,包括修改CMakeLists.txt文件以包含自定义消息结构体,并更新XML文件。

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

一、发布自带的库里面的msg

此时我们只需要创建发布对象;

ros::Publisher 自定义发布对象

ros::Publisher pub = n.advertise<std_msgs::String>("topic",10);

<std::msgs::String>表示范型;

topic:自定义

10:表示最多缓存10条,增加一条消息要删除前面的一条

编写发布逻辑;

包名::结构体名 实例化结构体对象

设置发布频率;

为了验证发布是否正确,用ROS_INFO();将发布的消息打印出来;

以下是发布的std_msgs::String 类型的消息的示例:

#include"ros/ros.h"
#include"std_msgs/String.h"


int main(int argc,char ** argv)
{
    //设置编码,防止中文出现乱码
    setlocale(LC_ALL,"");

    //初始化节点
    ros::init(argc,argv,"haha01");
    //初始化节点句柄
    ros::NodeHandle n;
    //创建发布对象
    ros::Publisher pub = n.advertise<std_msgs::String>("fang",10);
    //编写发布逻辑
    std_msgs::String msg;

    //设置发布频率,1表示1秒发布一次
    ros::Rate rate(1);
    int count = 0;

    while(ros::ok())
    {
        count++;
        //使用std::stringstream 拼接字符串与编号
        std::stringstream ss;
        ss <<"hello"<< count;
        msg.data = ss.str();
        //发布消息
        pub.publish(msg);
        //加入调试,打印消息
        ROS_INFO("发布的数据是: %s",ss.str().c_str());
        rate.sleep();
    }
    //ROS_INFO("hello world");
    return 0;

}

二、订阅发布的消息

创建订阅对象;

编写回调函数(回调函数是为了处理订阅到的数据);

以下是订阅刚才发布的消息的示例:ConstPtr&表示拿到的消息的常量指针的引用

#include"ros/ros.h"
#include"std_msgs/String.h"

void doMsg(const std_msgs::String::ConstPtr& msg_p)
{
    ROS_INFO("我听见: %s",msg_p->data.c_str());
}

int main(int argc,char ** argv)
{

    setlocale(LC_ALL,"");
    ros::init(argc,argv,"hear01");
    ros::NodeHandle n;
    ros::Subscriber sub = n.subscribe<std_msgs::String>("fang",10,doMsg);
    ros::spin();
    return 0;
}

CMakeLists修改:

我是把发布方和订阅方都放在了一个src下,所以下图对应的add_executable里面的节点名字要改成自己定义的,否则就会报错,因为如果不改的话,就会生成以包名开头的节点,而又添加了其他的可执行文件,就会发生冲突。

add_executable(${PROJECT_NAME}_node src/haha01.cpp  src/hear01.cpp  src/demo03.cpp src/demo04.cpp  )

如果写成该形式,则会报main函数多次定义;

 xml文件不用修改;

三、发布自定义的msg

首先要在包下创建一个msg文件夹,然后创建自定义的msg结构体,下图是本人创建的结构体示例:

 

此时我们需要创建发布对象;

编写发布逻辑;

设置发布频率;

!头文件要包含刚才的结构体,采用#include"包名/结构体.h"

发布自定义msg示例如下,此时没有打印输出发布的消息,可以自行加入,方法同上:

#include"ros/ros.h"
#include"std_msgs/String.h"
#include"bag/person.h"

int main(int argc,char **argv)
{
    setlocale(LC_ALL,"");
    ros::init(argc,argv,"demo03");
    ros::NodeHandle n;
    ros::Publisher pub = n.advertise<bag::person>("chatter",10);
    bag::person per;
    per.name = "zhangsan";
    per.age = 16;
    per.height = 178;
    ros::Rate rate(1);
    while (ros::ok())
    {
        pub.publish(per);
        rate.sleep();
        /* code */
    }
    return 0;
}

四、订阅发布的自定义msg

创建订阅对象

创建订阅对象的时候,我们可以不写范型,直接写topic,它会自动搜索;

ros::Subscriber sub = n.subscribe("chatter",10,doperson);

编写回调函数

以下是订阅上面发布的自定义消息的示例:

#include"ros/ros.h"
#include"std_msgs/String.h"
#include"bag/person.h"

void doperson(const bag::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::init(argc,argv,"demo04");
    ros::NodeHandle n;
    ros::Subscriber sub = n.subscribe("chatter",10,doperson);

    ros::spin();
    return 0;
}

因为有自定义msg,所以CMakeLists需要进行更多的配置,

 

同样xml文件也要进行简单的修改:

完成上述配置之后就可以编译了,然后运行相应的节点就可以查看效果了。 

<think>嗯,用户想了解如何在ROS中使用自定义消息类型进行话题通信。我之前整理过一些关于ROS话题通信的引用资料,现在需要结合这些资料来回答。 首先,根据引用[1],自定义msg的流程包括创建msg文件、配置package.xmlCMakeLists.txt,然后是编写发布订阅方的代码。引用[2]提到msg文件中可以使用Header类型,所以可能需要包含Header标头。引用[3]强调了发布方、接收方数据(自定义消息)三个关键点,以及vscode的配置。 用户的问题需要分步骤的教程。应该先讲创建msg文件的步骤,包括在功能包下新建msg目录,编写msg文件的内容,比如加入Header其他字段。接着是配置package.xml,添加对message_generationmessage_runtime的依赖。然后配置CMakeLists.txt,添加msg文件,生成消息代码。 然后需要说明如何编写发布订阅者的Python代码,注意导入自定义消息类型。同时要提醒用户可能遇到的路径问题,比如需要source devel/setup.bash,或者配置vscode的环境。这部分可以引用[3]提到的vscode配置注意事项。 最后,测试部分需要编译运行节点,使用rostopic命令查看消息是否正确发布。可能还要提到常见错误,比如消息未生成导致导入失败,或者环境变量未设置正确。 生成的相关问题应该围绕步骤展开,比如如何定义复杂消息,如何处理依赖,以及调试方法。这样用户可以根据问题进一步深入学习。</think>### ROS自定义消息实现话题通信教程 #### 一、创建自定义消息 1. 在功能包目录下新建`msg`文件夹,创建`Person.msg`文件: ```msg Header header string name uint8 age float32 height ``` 注意:第一行的`Header`类型需要包含`std_msgs/Header`声明[^2] #### 二、配置编译规则 1. 修改`package.xml`添加依赖: ```xml <build_depend>message_generation</build_depend> <exec_depend>message_runtime</exec_depend> ``` 2. 修改`CMakeLists.txt`: ```cmake find_package(catkin REQUIRED COMPONENTS roscpp std_msgs message_generation ) add_message_files( FILES Person.msg ) generate_messages( DEPENDENCIES std_msgs ) catkin_package( CATKIN_DEPENDS message_runtime ) ``` #### 三、编写发布者节点(publisher.py) ```python #!/usr/bin/env python import rospy from std_msgs.msg import Header from your_pkg.msg import Person # 替换为实际包名 def talker(): pub = rospy.Publisher('person_info', Person, queue_size=10) rospy.init_node('person_publisher', anonymous=True) rate = rospy.Rate(1) # 1Hz while not rospy.is_shutdown(): msg = Person() msg.header.stamp = rospy.Time.now() msg.name = "Alice" msg.age = 25 msg.height = 1.68 pub.publish(msg) rate.sleep() if __name__ == '__main__': try: talker() except rospy.ROSInterruptException: pass ``` #### 四、编写订阅者节点(subscriber.py) ```python #!/usr/bin/env python import rospy from your_pkg.msg import Person # 替换为实际包名 def callback(data): rospy.loginfo(f"Received: {data.name}, Age: {data.age}, Height: {data.height}") def listener(): rospy.init_node('person_subscriber') rospy.Subscriber("person_info", Person, callback) rospy.spin() if __name__ == '__main__': listener() ``` #### 五、编译与测试 1. 执行编译: ```bash catkin_make source devel/setup.bash ``` 2. 运行测试: ```bash # 终端1 roscore # 终端2 rosrun your_pkg publisher.py # 终端3 rosrun your_pkg subscriber.py # 终端4 rostopic echo /person_info ``` #### 六、常见问题处理 1. **消息未生成**:检查`CMakeLists.txt`的`generate_messages()`是否配置正确 2. **导入失败**:确认已执行`source devel/setup.bash`[^3] 3. **字段不匹配**:修改msg文件后需要重新编译
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值