ros3

简介

机器人是一种高度复杂的系统性实现,在机器人上可能集成各种传感器(雷达、摄像头、GPS…)以及运动控制实现,为了解耦合,在ROS中每一个功能点都是一个单独的进程,每一个进程都是独立运行的。更确切的讲,ROS是进程(也称为Nodes)的分布式框架。 因为这些进程甚至还可分布于不同主机,不同主机协同工作,从而分散计算压力。我们就需要ROS中的通信机制来实现不同进程间的数据交换。

ROS 中的基本通信机制主要有如下三种实现策略:

  • 话题通信(发布订阅模式)
  • 服务通信(请求响应模式)
  • 参数服务器(参数共享模式)

话题通信

概述

话题通信是ROS中使用最频繁的一种通信机制,它是基于发布-订阅模式的一种通信机制,某个节点向话题发布消息时,其他订阅该话题的节点就会收到这条消息。话题通信一般用于不断更新的、少逻辑处理的数据传输场景,应用场景也十分广泛。

机器人在执行导航功能,使用的传感器是激光雷达,机器人会采集激光雷达感知到的信息并计算,然后生成运动控制信息驱动机器人底盘运动。

在上述场景中,就使用到了话题通信。

  • 以激光雷达信息的采集处理为例,在 ROS 中有一个节点需要时时的发布当前雷达采集到的数据,导航模块中也有节点会订阅并解析雷达数据。
  • 再以运动消息的发布为例,导航模块会根据传感器采集的数据时时的计算出运动控制信息并发布给底盘,底盘也可以有一个节点订阅运动信息并最终转换成控制电机的脉冲信号。

以此类推,像雷达、摄像头、GPS…. 等等一些传感器数据的采集,也都是使用了话题通信,换言之,话题通信适用于不断更新的数据传输相关的应用场景。

理论模型

img

角色

  • master -> 管理者
  • talker -> 发布者
  • listener -> 订阅者

流程

master根据话题建立发布者和订阅者之间的连接。

​ 0、talker在master中进行注册,提交话题以及RPC地址。

​ 1、listener在master中注册关注的话题。

​ 2、master将话题的RPC地址发送给listener

​ 3、listener根据地址访问talker

​ 4、talker响应自己的TCP地址

​ 5、listener连接talker的TCP服务

​ 6、talker与listener建立连接,talker发布消息,listener接收消息

注意事项

  • 使用RPC与TCP两种通信方式;

  • talker注册发布话题与listener关注话题没有先后顺序(步骤0与步骤1);

  • talker与listener可以存在多个;

  • talker与listener一旦建立TCP连接,master就可以退出;

  • 该实现模型在ROS已经封装

基本操作

编写发布订阅实现,要求发布方以10HZ(每秒10次)的频率发布文本消息并添加编号,订阅方订阅消息并将消息内容打印输出。

之后的项目学习均采用vscode编写,有关vscode的安装、配置以及使用参考ROS集成开发环境搭建

C++实现

实现流程
  • 发布方实现
  • 订阅方实现
  • 编辑配置文件
  • 编译执行
发布方实现
  • 新建cpp文件
  • 引入头文件
  • 初始化ROS节点
  • 创建节点句柄
  • 创建发布者对象
  • 编写发布逻辑并发布数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#include "ros/ros.h"
// ROS 文本类型库 std_msgs/String.h
#include "std_msgs/String.h"

// 用于字符拼接
#include <sstream>

/*
发布方实现
*/

int main(int argc, char *argv[])
{
// 解决中文乱码
setlocale(LC_ALL,"");
// 初始化ROS节点
ros::init(argc, argv, "pub_client");

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

// 创建发布者对象
ros::Publisher pub = nh.advertise<std_msgs::String>("test", 10);

// 编写发布逻辑并发布数据

// 创建发布消息
std_msgs::String msg;

// 设置发布频率
ros::Rate rate(10);

// 设置编号计数器

int count = 0;

// 循环发布
while (ros::ok())
{
// 实现拼接
std::stringstream ss;

ss<< "hello --> "<<++count;

msg.data = ss.str();

// 发布
pub.publish(msg);

// 日志
ROS_INFO("发布数据:%s",ss.str().c_str());

rate.sleep();
}

return 0;
}

接下来就是编辑配置文件,编译执行。

image-20220629165108494

可以在新的终端窗口使用rostopic echo 话题名命令查看话题发布结果。

image-20220629163812816

订阅方实现
  • 新建cpp文件
  • 引入头文件
  • 初始化ROS节点
  • 创建节点句柄
  • 创建订阅对象
  • 消息回调处理
  • ros::spin() 跳回处理回调函数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
#include "ros/ros.h"
// ROS 文本类型库 std_msgs/String.h
#include "std_msgs/String.h"

/*
订阅方实现
*/

void callback(const std_msgs::String::ConstPtr &msg){

//消息回调处理
ROS_INFO("订阅者接收到数据:%s",msg->data.c_str());
}

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

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

// 创建订阅者对象
ros::Subscriber sub = nh.subscribe("test", 10, callback);

// 跳回处理回调函数
ros::spin();

return 0;
}

编辑配置文件,编译执行。

image-20220629171553158

Python实现

实现流程
  • 发布方实现
  • 订阅方实现
  • python添加可执行权限(chmod)
  • 编辑配置文件
  • 编译执行
发布方实现
  • 新建Python文件
  • 导入消息类型包
  • 初始化ROS节点
  • 创建发布者对象
  • 编写发布逻辑并发布数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#! /usr/bin/env python
import rospy
# 发布消息类型
from std_msgs.msg import String

if __name__ == "__main__":
# 初始化ROS节点
rospy.init_node("pub_client")

# 创建发布者对象
pub = rospy.Publisher("test", String, queue_size = 10)

# 创建数据
msg = String()

# 设置发布频率
rate = rospy.Rate(1)

# 设置计数器
count = 0
# 循环发布
while not rospy.is_shutdown():
count += 1
msg.data = "hello -> {}".format(str(count))

# 发布数据
pub.publish(msg)

# 打印日志
rospy.loginfo("发布的数据是:%s",msg.data)

rate.sleep()

添加可执行权限、编辑配置文件,执行。

image-20220629174330521

订阅方实现
  • 新建Python文件

  • 导入消息类型包

  • 初始化ROS节点

  • 创建订阅对象

  • 消息回调处理

  • rospy.spin()跳回处理回调函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#! /usr/bin/env python
import rospy
# 导入发布消息类型包
from std_msgs.msg import String

def callback(msg):
# 消息回调
rospy.loginfo("接收的数据是:%s", msg.data)

if __name__ == "__main__":
# 初始化 ROS 节点
rospy.init_node("sub_client")

# 创建订阅者对象
sub = rospy.Subscriber("test", String, callback, queue_size=10)

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

添加可执行权限、编辑配置文件,执行。

image-20220629181216820

注意事项

  • vscode 中的 main 函数 声明 int main(int argc, char const *argv[]){},默认生成 argv 被 const 修饰,需要去除该修饰符

  • 订阅时,第一条数据丢失

    原因: 发送第一条数据时, publisher 还未在 roscore 注册完毕

    解决: 注册后,加入休眠 延迟第一条数据的发送。

    C++: ros::Duration(3.0).sleep();

    Python:rospy.sleep(3)

自定义消息

之前我们使用了ROS自带的字符串类型消息进行话题通信,除了字符串以外,ROS 中还 封装了一些原生的数据类型,比如:String、Int32、Int64、Char、Bool等,但在实际应用中,数据往往都是多种类型的组合体,因此需要自定义的消息类型来满足实际的应用场景。下面将开始学习自定义消息类型,并使用自定义的消息类型收发数据。

自定义消息具体的实现流程如下:

  • 按照固定格式创建 msg 文件

  • 编辑配置文件

  • 编译生成可以被 Python 或 C++ 调用的中间文件

定义msg文件

在功能包下新建msg目录,添加Person.msg文件

1
2
3
string name
uint16 age
float64 height

编辑配置文件

package.xml

在package.xml中添加以下两条配置

1
2
<build_depend>message_generation</build_depend>
<exec_depend>message_runtime</exec_depend>
CMakeLists.txt

在find_package中添加message_generation

1
2
3
4
5
6
find_package(catkin REQUIRED COMPONENTS
roscpp
rospy
std_msgs
message_generation
)

在add_message_files中加入Person.msg

1
2
3
4
add_message_files(
FILES
Person.msg
)

添加generate_message配置

1
2
3
4
generate_messages(
DEPENDENCIES
std_msgs
)

catkin_package中添加依赖

1
2
3
4
5
6
catkin_package(
# INCLUDE_DIRS include
# LIBRARIES plumbing_pub_sub
CATKIN_DEPENDS roscpp rospy std_msgs message_runtime
# DEPENDS system_lib
)

编译

编译后会生成消息调用的中间文件

C++:.../工作空间/devel/include/包名/xxx.h

Python:.../工作空间/devel/lib/python3/dist-packages/包名/msg

image-20220630122945978

调用

vscode的配置

为了方便代码提示以及误抛异常,需要先配置 vscode。

  • C++

    在c_cpp_properties.json 的 includepath中加入刚刚编译生成的头文件路径

    image-20220630123729889
  • Python

    在settings.json的python.autoComplete.extraPaths中配置刚刚编译生成的包路径

    image-20220630123950752
导入头文件/包
  • C++

    1
    #include "包名/Person.h"
  • Python

    1
    from 包名.msg import Person

导入之后Person就可以作为一种消息类型在通信中使用了。