前言:作为目前最广泛使用的开源机器人软件平台,ROS(Robot Operating System)极大的提升了机器人开发的效率。目前无人驾驶系统的实际实现仍然依赖于ROS,作为一个相对成熟的机器人软件平台,ROS虽然存在一些缺点,在无人驾驶系统的系统实现上,ROS仍然是开源框架中最好的选择,这也是为什么诸如Apollo,Autoware等无人驾驶开源项目会选择ROS。本文主要讨论ROS的入门知识,后续的文章会对基于ROS的无人车系统编程进行进一步探讨。
ROS概念:ROS = Robot Operating System,包括:
1. 通道:ROS提供了一种发布-订阅式的通信框架用以简单、快速地构建分布式计算系
2. 工具:ROS提供了大量的工具组合用以配置、启动、自检、调试、可视化、登录、测试、终止分布式计算系统;
3. 能力:具有控制,规划,预测,定位操纵等功能;
4. 生态系统:ROS的支持与发展依托着一个强大的社区。ros.org尤其关注兼容性和支持文档,提供了一套“一站式”的方案使得用户得以搜索并学习来自全球开发者数以千计的ROS程序包。
ROS系统是起源于2007年斯坦福大学人工智能实验室的STAIR项目与机器人技术公司Willow Garage的个人机器人项目(Personal Robots Program)之间的合作,2008年之后就由Willow Garage来进行推动。如今已被许多的学校公司所使用。为机器人变成提供了实际标准(Defacto standard)
需要理解的ROS中的核心概念包括:
用于管理节点间的通信,ROS Master 通过RPC(Remote Procedure Call Protocol,远程过程调用)提供了登记列表和对其他计算图表的查找。没有控制器,节点将无法找到其他节点,交换消息或调用服务。
开启Master命令:
roscore
节点就是一些独立编译,执行运算任务的进程。ROS利用规模可增长的方式使代码模块化:一个系统就是典型的由很多节点组成的。在这里,节点也可以被称之为“软件模块”。我们使用“节点”使得基于ROS的系统在运行的时候更加形象化:当许多节点同时运行时,可以很方便的将端对端的通讯绘制成一个图表,在这个图表中,进程就是图中的节点,而端对端的连接关系就是其中弧线连接。
运行节点:
rosrun package_name node_name
查看激活节点列表:
rosnode list
检索有关节点的信息:
rosnode info node_name
在ROS中消息以一种发布/订阅的方式传递。一个节点可以在一个给定的主题中发布消息。一个节点针对某个主题关注与订阅特定类型的数据。可能同时有多个节点发布或者订阅同一个主题的消息。总体上,发布者和订阅者不了解彼此的存在。
查看激活主题:
rostopic list
订阅和打印一个主题的内容:
rostopic echo /topic
显示主题相关信息:
rostopic info /topic
消息用来定义主题topic类型的数据结构,是整型,浮点型,布尔型,字符串等一系列结构进行压缩后,定义成*.msg文件。
查看主题类型:
rostopic type /topic
向topic发布一个消息:
rostopic pub /topic type args
catkin 创建系统
catkin是ROS构建系统,用来生成可执行文件,库和接口。
(现在很多人建议用catkin build 来代替 catkin_make)
首先进入工作区:
cd ~/catkin_ws
然后创建一个包:
catkin_create_pkg package_name depend1 depend2 depend3
每当你创建一个新包时,都需要更新你的环境:
source devel/setup.zsh
在catkin创建的工作区中包含以下三个文件夹:
1. src:这个就是用来存放源码的地方,操作者在这里面进行编码,修改
2. build:用来存放缓存文件和一些中间文件
3. devel:(development)存放创建目标的地方(在存放之前安装)
ROS的软件使用package(包)进行组织,包下通常包含一下内容:
这个文件定义了package的属性,这些属性包括:
下面是一个简单的例子的package信息:
<package format="2">
<name>ros_practicename>
<version>0.0.1version>
<description>The ros_practice packagedescription>
<maintainer email="[email protected]">Adammaintainer>
<license>MITlicense>
<buildtool_depend>catkinbuildtool_depend>
<build_depend>roscppbuild_depend>
<build_depend>sensor_msgbuild_depend>
<build_export_depend>roscppbuild_export_depend>
<build_export_depend>sensor_msgbuild_export_depend>
<exec_depend>roscppexec_depend>
<exec_depend>sensor_msgexec_depend>
package>
在这个例子中,我们使用了roscpp
的client的库,同时我们使用了一个sensor_msg
消息。
CMakeLists.txt文件是CMake构建系统的输入,在这里我们不会详细讨论CMake的写法(因为它本身可以很复杂),我们大致熟悉一下我们常用的CMake的语法:
cmake_minimum_required
:需要的CMake的最低版本project()
:包的名称find_package()
查找建构是需要的其他 CMake/Catkin 包add_message_files()
add_service_files()
add_action_files
生成Message/Service/Actiongenerate_messages()
调用消息生成catkin_package()
指定包的构建信息add_library()/add_executable()/target_link_libraries()
用于构建的库,可执行代码install()
安装规则下图是一个 CMakeLists.txt
的例子:
我们使用Husky模拟器来实践ROS编程,Husky本身是一种小型的轮式移动机器人(和我们关注的无人车比较相似),该模拟器基于Gazebo物理仿真环境,Gazebo是一款机器人仿真软件,基于ODE物理引擎,可以模拟机器人以及环境中的很多物理特性,Husky模拟器的安装请参考:
http://wiki.ros.org/husky_gazebo/Tutorials/Simulating%20Husky
如果ros kinetic使用apt安装存在问题的话,请参考:https://answers.ros.org/question/256756/how-to-install-husky-simulator-in-kinetic/
使用roslaunch启动husky模拟器,启动一个空的世界:
roslaunch husky_gazebo husky_empty_world.launch
roslaunch是ROS中启动多个节点的机制,Launch文件是ROS提供的可以同时运行多个nodes的文件。Launch文件以一种特殊的XML格式编写,以.launch结尾。
在运行了模拟器以后使用rqt_graph
查看正在运行的节点和话题:
我们发现/gazebo
节点订阅了一个/cmd_vel
话题,使用rostopic info /husky_velocity_controller/cmd_vel
查看话题信息,得到如下信息:
Type: geometry_msgs/Twist
Publishers: * /twist_mux (http://adam:40181/) Subscribers: * /gazebo (http://adam:35678/)
也就是说,这个控制指令是一个geometry_msgs/Twist
消息,ROS中我们使用geometry_msgs/Twist
消息类型来发布运动命令。控制命令会被基本的控制器节点使用。话题 /cmd_vel
全名 ” command velocities “ 。控制器节点订阅 /cmd_vel
话题,并将里面的运动命令(Twist 消息)通过 PID 控制算法转换成电机信号。使用osmsg show geometry_msgs/Twist
查看消息的格式:
geometry_msgs/Vector3 linear
float64 x
float64 y
float64 z
geometry_msgs/Vector3 angular
float64 x
float64 y
float64 z
其中,linear
表示线速度(单位:m/s) ;angular
表示角速度(单位:rad/s) 。
下面我们在指令行中通过向/husky_velocity_controller/cmd_vel
发送Twist消息来驱动我们的小车,我们让它以0.2m/s的速度和0.2rad/s的角速度运动:
rostopic pub -r 10 /husky_velocity_controller/cmd_vel geometry_msgs/Twist '{linear: {x: 4, y: 0, z: 0}, angular: {x: 0, y: 0, z: 0.5}}'
其实我们还可以直接向/cmd_vel
主题发消息来控制小车(不信的话可以试试rostopic info /cmd_vel
看看订阅它的节点):
rostopic pub -r 10 /cmd_vel geometry_msgs/Twist '{linear: {x: 4, y: 0, z: 0}, angular: {x: 0, y: 0, z: 0.5}}'
接下来我们使用手柄来控制Husky机器人,我们使用teleop_twist_joy
包来实现对手柄数据转换为Twist指令并将Twist发布到/cmd_vel
主题,启动teleop_twist_joy
下的launch文件:
roslaunch teleop_twist_joy teleop.launch
启动后会在终端打印如下的信息:
PARAMETERS
* /joy_node/autorepeat_rate: 20
* /joy_node/deadzone: 0.3
* /joy_node/dev: /dev/input/js0
* /rosdistro: kinetic
* /rosversion: 1.12.12
* /teleop_twist_joy/axis_angular: 0
* /teleop_twist_joy/axis_linear: 1
* /teleop_twist_joy/enable_button: 8
* /teleop_twist_joy/enable_turbo_button: 10
* /teleop_twist_joy/scale_angular: 0.4
* /teleop_twist_joy/scale_linear: 0.7
* /teleop_twist_joy/scale_linear_turbo: 1.5
NODES
/
joy_node (joy/joy_node)
teleop_twist_joy (teleop_twist_joy/teleop_node)
下面的NODES表示启动了joy_node
和teleop_node
两个节点,它们分别位于joy
和teleop_twist_joy
两个包中,上面是启动文件的参数信息,我们着重看一下三个参数:
* /teleop_twist_joy/axis_angular: 0
* /teleop_twist_joy/axis_linear: 1
* /teleop_twist_joy/enable_button: 8
其中enable_button
表示必须要按下该按键(8号按键,在我的手柄上是back键)才能控制,前面两个分别表示控制速度和角速度的按钮(在我的手柄上是左摇杆)
Note: Husky作为一种轮式机器人,其能够控制的速度只有x方向的速度(前向)和z方向的角速度(Yaw,偏航角速度)
我们看一下此时的rqt_graph:
至此,命令行下的ros操作相信大家都熟悉并理解了,后面我们开始ROS编程部分。