[ROS2实战] 从零打造SLAM机器人(一):基于ESP32与Micro-ROS的底盘运动控制与里程计实现
1. 前言:为什么SLAM需要一个好底盘?
- SLAM的本质:SLAM 就像人闭着眼睛走路摸墙画地图。
- 底盘的作用:
- 如果人走路晃晃悠悠(电机控制不稳),地图就会画歪。
- 如果人不知道自己迈了几步(里程计不准),地图就会断裂。
- 本篇目标:抛弃昂贵的工控机,用几十块钱的 ESP32 单片机,通过 Micro-ROS 接入 ROS 2 生态,实现精准的 PID闭环控制 和 里程计(Odometry)发布,为后续接入激光雷达做准备。
2. 系统架构设计
- 上位机:Ubuntu 22.04 (虚拟机) + ROS 2 Humble + Micro-ROS Agent。
- 下位机:ESP32 开发板 + 直流减速电机 + 霍尔编码器。
- 通信链路:
- 物理层:Wi-Fi (UDP协议),彻底摆脱 USB 线束缚。
- 协议层:Micro-ROS (XRCE-DDS),让单片机成为 ROS 2 网络中的一等公民。
3. 核心技术点复盘(这是干货部分)
3.1 环境搭建与“填坑”
- Docker Agent:使用 Docker 运行 Micro-ROS Agent,避免了复杂的环境配置。
- PlatformIO:使用 VS Code + PlatformIO 进行嵌入式开发,利用
lib目录管理多模块代码。 - 网络隔离之战(重点):
- 问题:虚拟机 NAT 模式下,外部 ESP32 无法连接内部 Agent;桥接模式又导致虚拟机崩溃。
- 绝杀方案:NAT模式 + 端口转发 (Port Forwarding)。
- 操作:将主机的 UDP 8888 映射到虚拟机的 UDP 8888,ESP32 直连 Windows 主机 IP,实现数据“穿墙”。
3.2 闭环控制:PID 算法
- 为什么要 PID:开环控制(直接给占空比)受电量、摩擦力影响大,速度不可控。
- 实现:
- 测量:利用 ESP32 的 PCNT 硬件单元读取编码器脉冲。
- 计算:
目标速度 - 实际速度 = 误差,通过 PID 公式计算出补偿后的 PWM。 - 效果:无论负载变化,轮子都能死死咬住目标速度(例如 20mm/s)。
3.3 机器人的“小脑”:两轮差速运动学
- 运动学逆解 (Inverse Kinematics) —— 听指挥:
- ROS 发来
/cmd_vel($v, \omega$) $\rightarrow$ 算出左右轮目标转速 。
- ROS 发来
- 运动学正解 (Forward Kinematics) —— 自省:
- 读取左右轮实际速度 推算出机器人整体速度。
3.4 机器人的“自我感知”:里程计 (Odometry)
- 原理:航位推算(Dead Reckoning)。
- 实现:
- 利用正解算出的速度,乘以微小时间间隔 $dt$。
- 累加计算出全局坐标和朝向。
- 坐标系转换:
- 将欧拉角 (Yaw) 转换为 ROS 2 标准的 四元数 (Quaternion) 发布。
- 发布 TF 变换(虽然本例暂时只发了话题,但在 SLAM 中 TF 是必须的)。
4. 关键代码片段展示
Kinematics::kinematic_inverse(逆解实现)Kinematics::update_bot_odom(里程计积分实现)microros_task中的 UDP 连接配置。
5. 联调与可视化验证
- 工具:Rviz2。
- 遇到的坑与解决:
- 现象:Rviz2 收不到数据,或者不显示箭头。
- 原因1:QoS 不匹配。Micro-ROS 默认 Best Effort,Rviz2 默认 Reliable。必须修改 Rviz2 配置为 Best Effort。
- 原因2:时间同步。ESP32 必须调用
rmw_uros_sync_session与电脑对表,否则数据会被视为“过期”丢弃。 - 原因3:坐标系。Fixed Frame 必须设置为
odom。
- 最终效果:
- 启动
teleop_twist_keyboard控制小车。 - Rviz2 中的红色箭头随着键盘指令同步移动和旋转,且数据平滑。
- 启动
6. 总结与展望
- 当前状态:我们要到了一个能动、能听、知道自己在哪的“瞎子”机器人。
- 下一步计划:
- 加眼睛:接入激光雷达(Lidar)。
- 长脑子:运行 Cartographer 或 Gmapping 算法。
- 最终目标:看着它在房间里自动构建出户型图!
给读者的建议
- 供电很重要:Wi-Fi 模式耗电大,USB 供电不足会导致 Micro-ROS 频繁断连,建议使用电池供电。
- 善用 Git:每完成一个功能模块(PID、正解、逆解)就 commit 一次,方便回溯。
- 网络是最大的坑:如果是在虚拟机里开发,搞定网络设置就成功了一半。