ROS学习记录(四)——话题消息的定义与使用

ROS学习记录(一)——基本概念
ROS学习记录(二)——创建功能包
ROS学习记录(三)——Publisher和Subscriber编程实现
ROS学习记录(四)——话题消息的定义与使用
ROS学习记录(五)——Client和Server编程实现

平台:Ubuntu18.04,ROS-melodic
实验目的:当ROS已定义好的消息类型不能满足自己的使用,我们就可以自行定义消息类型,来满足自己的需求。
实验模拟Publisher发布Person信息,Subscriber订阅Person信息,通过名为/person_info话题名通信。
框架图:
在这里插入图片描述

实验(C语言版本)

如何自定义消息类型:
在这里插入图片描述
步骤:

  1. cd ~/catkin_ws/src/test_pkg进入功能包存放代码的地方,mkdir msg&&cd msgtouch Person.msg
  2. 打开Person.msg文件,复制下方代码,粘贴进去,保存并关闭文件
string name
uint8 sex
uint8 age

uint8 unknown = 0
uint8 male = 1
uint8 female = 2
  1. 在test_pkg/package.xml中添加功能包依赖,复制下方代码,粘贴在依赖项的最下方
<build_depend>message_generation</build_depend>
<exec_depend>message_runtime</exec_depend>

package.xml文件包含了如:功能包名、版本号、功能包描述、功能包编译时的依赖库信息
这两个一个是编译功能包依赖,一个是执行功能包依赖。

  1. 在CMakeLists.txt添加编译选项
find_package( …… message_generation)
add_message_files(FILES Person.msg)
generate_messages(DEPENDENCIES std_msgs)
catkin_package(…… message_runtime)

第一行,因为刚刚添加了一个编译功能包依赖,所以在find_package()中加入message_generation
第二、三行,直接复制粘贴进去(在合适的位置)
第四行,因为刚刚添加里一个执行功能包依赖,所以在catkin_package()中加入message_runtime

  1. cd ~/catkin_wscatkin_make等待编译完成。
  2. cd src/test_pkg/srctouch person_publisher.cpptouch peron_subscriber.cpp。将代码粘贴进去。
    person_publisher.cpp:
/**
 * 该例程将发布/person_info话题,自定义消息类型learning_topic::Person
 */
 
#include <ros/ros.h>
#include "test_pkg/Person.h"   //!!!!!!!!!!!!

int main(int argc, char **argv)
{
    // ROS节点初始化
    ros::init(argc, argv, "person_publisher");

    // 创建节点句柄
    ros::NodeHandle n;

    // 创建一个Publisher,发布名为/person_info的topic,消息类型为test_pkg::Person,队列长度10
    ros::Publisher person_info_pub = n.advertise<test_pkg::Person>("/person_info", 10);
    // test_pkg::Person数据类型是通过头文件载入的


    // 设置循环的频率
    ros::Rate loop_rate(1);

    int count = 0;
    while (ros::ok())
    {
        // 初始化learning_topic::Person类型的消息
    	test_pkg::Person person_msg;
		person_msg.name = "Tom";
		person_msg.age  = 18;
		person_msg.sex  = test_pkg::Person::male;  
		//通过这句可调用在Person.msg里定义的宏

        // 发布消息
		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:

/**
 * 该例程将订阅/person_info话题,自定义消息类型learning_topic::Person
 */
 
#include <ros/ros.h>
#include "test_pkg/Person.h"

// 接收到订阅的消息后,会进入消息回调函数
void personInfoCallback(const test_pkg::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;
}

  1. 在CMakeLists.txt中添加下方代码
###########
## Build ##
###########

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_publisher ${PROJECT_NAME}_generate_messages_cpp)

add_dependencies() 的作用是动态的跟刚刚生成的Person头文件产生依赖关系。因为有些代码是动态生成的,所以需要让可执行文件和动态生成的程序产生依赖关系。即涉及到自定义类型消息的时候,必须要有这句代码。
注意,如果不加上述6行代码,可以编译通过,没有报错,但实际上是没有编译person_publisher.cpp和person_subscriber.cpp,也就是用不了。

  1. 执行cd ~/catkin_wscatkin_make等待编译完成。
  2. 打开一个新终端,启动ROS masterroscore,再打开一个新终端,执行rosrun test_pkg person_subscriber,再打开一个新终端,执行rosrun test_pkg person_publisher
  3. publisher实时发送,subscriber实时接收打印,成功。

此时如果关闭roscore终端,person_subscriber和person_publisher通信继续,这是因为ROS Master相当于“婚介中心”,一旦“婚介”成功,它的存在对于两者普通的通信就不重要了,可以关闭。

实验(python语言版本)

注意点:在使用python语言版本的程序时,程序脚本需要有可执行权限,可以使用chomd +x 文件名赋予可执行权限

使用python脚本,只需要将.py脚本拷贝到test_pkg/src中,给予可执行权限,直接用rosrun test_pkg 脚本.py即可。

person_publisher.py:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# 该例程将发布/person_info话题,自定义消息类型test_pkg::Person

import rospy
from test_pkg.msg import Person

def velocity_publisher():
	# ROS节点初始化
    rospy.init_node('person_publisher', anonymous=True)

	# 创建一个Publisher,发布名为/person_info的topic,消息类型为test_pkg::Person,队列长度10
    person_info_pub = rospy.Publisher('/person_info', Person, queue_size=10)

	#设置循环的频率
    rate = rospy.Rate(10) 

    while not rospy.is_shutdown():
		# 初始化learning_topic::Person类型的消息
    	person_msg = Person()
    	person_msg.name = "Tom";
    	person_msg.age  = 18;
    	person_msg.sex  = Person.male;

		# 发布消息
        person_info_pub.publish(person_msg)
    	rospy.loginfo("Publsh person message[%s, %d, %d]", 
				person_msg.name, person_msg.age, person_msg.sex)

		# 按照循环频率延时
        rate.sleep()

if __name__ == '__main__':
    try:
        velocity_publisher()
    except rospy.ROSInterruptException:
        pass

person_subscriber.py:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# 该例程将订阅/person_info话题,自定义消息类型learning_topic::Person

import rospy
from test_pkg.msg import Person

def personInfoCallback(msg):
    rospy.loginfo("Subcribe Person Info: name:%s  age:%d  sex:%d", 
			 msg.name, msg.age, msg.sex)

def person_subscriber():
	# ROS节点初始化
    rospy.init_node('person_subscriber', anonymous=True)

	# 创建一个Subscriber,订阅名为/person_info的topic,话题的消息类型是Person,注册回调函数personInfoCallback
    rospy.Subscriber("/person_info", Person, personInfoCallback)

	# 循环等待回调函数
    rospy.spin()

if __name__ == '__main__':
    person_subscriber()
### 实现ROS 2话题接收转发 #### 安装必要的包 为了使 ROS 1 和 ROS 2 能够互相传递话题,需安装桥梁工具。对于 ROS Noetic 和 ROS Foxy 组合而言,应执行如下命令来完成 ros1_bridge 的安装: ```bash sudo apt-get install ros-foxy-ros1-bridge ``` 启动此桥接服务前,先初始化两个环境变量以便于同时支持两种版本的 ROS 命令集[^1]。 #### 配置环境并激活桥接功能 接着,在同一会话窗口内依次加载对应版本的设置脚本,并调用 `dynamic_bridge` 来自动创建所有活跃主题之间的映射关系: ```bash source /opt/ros/noetic/setup.bash source /opt/ros/foxy/setup.bash ros2 run ros1_bridge dynamic_bridge --bridge-all-topics ``` 这一步骤确保了来自任一系统的任何新出现的主题都能被即时识别并另一侧建立联系。 #### 编写Python代码用于订阅和发布话题 下面展示了一段 Python 代码片段,它展示了怎样在一个节点里既作为订阅者又扮演着发布者的双重身份工作。这段程序能够监听特定名称下的消息流并将收到的内容重新广播出去——可能是相同的或是不同的目标地址。 ```python import rclpy from rclpy.node import Node from std_msgs.msg import String class TopicRelay(Node): def __init__(node): super().__init__('topic_relay') node.subscription = node.create_subscription( String, 'input_topic', node.listener_callback, 10) node.publisher_ = node.create_publisher(String, 'output_topic', 10) def listener_callback(node, msg): node.get_logger().info('Received message: "%s"' % msg.data) output_msg = String() output_msg.data = msg.data node.publisher_.publish(output_msg) def main(args=None): rclpy.init(args=args) relay_node = TopicRelay() try: rclpy.spin(relay_node) except KeyboardInterrupt: pass relay_node.destroy_node() rclpy.shutdown() if __name__ == '__main__': main() ``` 上述例子中定义了一个名为 `TopicRelay` 的类继承自 `Node` 类型对象;构造函数内部设置了输入输出通道以及回调处理逻辑;当有新的字符串类型的数据到达时触发相应动作,即记录日志并向指定路径推送副本[^4]。 #### 测试配置有效性 最后可以通过图形界面工具 RQT 或其他方式验证整个链路是否正常运作。例如,启动RQT可以方便地监控当前活动中的各个组件状态变化情况: ```bash rqt ``` 一旦确认无误,则证明已经成功搭建起了从源到目的地的信息传输管道[^2]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值