《自动驾驶轨迹规划实战:Lattice Planner实现避障路径生成(附可运行Python代码)》—— 零基础实现基于离散优化的避障路径规划


《自动驾驶轨迹规划实战:Lattice Planner实现避障路径生成(附可运行Python代码)》
—— 零基础实现基于离散优化的避障路径规划


一、为什么Lattice Planner成为自动驾驶的核心算法?

在自动驾驶的路径规划领域,Lattice Planner(格子规划器)凭借其结构化采样+动态优化的特性,成为Waymo、Aptiv等头部企业的核心算法。与传统A*、RRT算法不同,Lattice Planner通过横向位移采样+纵向速度规划的复合策略,能够在10ms内生成满足车辆运动学约束的平滑轨迹。本文将手把手实现一个简化版Lattice Planner,并针对经典“避障场景”进行代码级解析。


二、算法原理精讲(附数学推导图示)

2.1 轨迹生成三要素
s(t) = s_0 + v_0 t + \frac{1}{2}a t^2 \quad \text{(纵向运动方程)}
d(s) = c_0 + c_1 s + c_2 s^2 + c_3 s^3 \quad \text{(横向多项式拟合)}
2.2 成本函数设计
评价维度权重系数计算公式
舒适性0.4∑(jerk²)
安全性0.3障碍物距离倒数
路径贴合度0.2参考线偏移量
能耗效率0.1速度变化率

三、Python代码实现(可直接运行)

3.1 环境初始化
import numpy as np
import matplotlib.pyplot as pltclass Vehicle:def __init__(self):self.width = 2.0    # 车宽self.length = 4.5   # 车长self.max_acc = 2.0  # 最大加速度(m/s²)self.max_jerk = 1.5 # 最大加加速度(m/s³)obstacles = [{'x': 30, 'y': 1.8, 'r': 1.2},{'x': 45, 'y': -1.5, 'r': 0.8}
]  # 障碍物坐标及半径
3.2 轨迹采样核心函数
def generate_lattice_paths(current_state, horizon=5.0, dt=0.1):""":param current_state: 当前状态 [s, s_dot, d, d_dot]:param horizon: 规划时长(s):return: 候选轨迹列表"""paths = []# 纵向速度采样(3种模式)for s_dot_end in np.linspace(8, 12, 3):  # 8m/s~12m/s# 横向位移采样(5种偏移)for d_end in np.linspace(-1.5, 1.5, 5):  t = np.arange(0, horizon + dt, dt)# 纵向四次多项式拟合s_coeff = np.polyfit([0, horizon], [current_state[0], current_state[0] + s_dot_end*horizon], 4)# 横向五次多项式拟合d_coeff = np.polyfit([0, horizon], [current_state[2], d_end], 5)path = {'s': np.polyval(s_coeff, t),'d': np.polyval(d_coeff, t),'s_dot': np.polyval(np.polyder(s_coeff), t),'d_dot': np.polyval(np.polyder(d_coeff), t)}paths.append(path)return paths
3.3 碰撞检测与最优选择
def select_optimal_path(paths, obstacles):min_cost = float('inf')best_path = Nonefor path in paths:# 计算舒适性成本jerk_cost = np.mean(np.diff(path['s_dot'])**2)# 计算安全成本collision_cost = 0for t in range(len(path['s'])):x = path['s'][t]y = path['d'][t]for obs in obstacles:dist = np.sqrt((x - obs['x'])**2 + (y - obs['y'])**2)if dist < (obs['r'] + 1.5):  # 1.5m为安全距离collision_cost += 1 / max(dist, 0.1)# 计算总成本total_cost = 0.4*jerk_cost + 0.3*collision_costif total_cost < min_cost:min_cost = total_costbest_path = pathreturn best_path
3.4 可视化运行
def plot_scenario(paths, best_path, obstacles):plt.figure(figsize=(10, 6))# 绘制障碍物for obs in obstacles:circle = plt.Circle((obs['x'], obs['y']), obs['r'], color='r', alpha=0.3)plt.gca().add_patch(circle)# 绘制所有候选路径for path in paths:plt.plot(path['s'], path['d'], 'gray', alpha=0.2)# 高亮最优路径plt.plot(best_path['s'], best_path['d'], 'b', linewidth=2)plt.xlabel('Longitudinal Position (m)')plt.ylabel('Lateral Position (m)')plt.grid(True)plt.axis('equal')plt.show()# 主程序
if __name__ == "__main__":ego_state = [20, 10, 0, 0]  # [s, s_dot, d, d_dot]candidate_paths = generate_lattice_paths(ego_state)optimal_path = select_optimal_path(candidate_paths, obstacles)plot_scenario(candidate_paths, optimal_path, obstacles)

完整代码

#!/usr/bin/python3
import numpy as np
import matplotlib.pyplot as pltclass Vehicle:def __init__(self):self.width = 2.0    # 车宽self.length = 4.5   # 车长self.max_acc = 2.0  # 最大加速度(m/s²)self.max_jerk = 1.5 # 最大加加速度(m/s³)obstacles = [{'x': 30, 'y': 1.8, 'r': 1.2},{'x': 45, 'y': -1.5, 'r': 0.8}
]  # 障碍物坐标及半径
def generate_lattice_paths(current_state, horizon=5.0, dt=0.1):""":param current_state: 当前状态 [s, s_dot, d, d_dot]:param horizon: 规划时长(s):return: 候选轨迹列表"""paths = []# 纵向速度采样(3种模式)for s_dot_end in np.linspace(8, 12, 3):  # 8m/s~12m/s# 横向位移采样(5种偏移)for d_end in np.linspace(-1.5, 1.5, 5):  t = np.arange(0, horizon + dt, dt)# 纵向四次多项式拟合s_coeff = np.polyfit([0, horizon], [current_state[0], current_state[0] + s_dot_end*horizon], 4)# 横向五次多项式拟合d_coeff = np.polyfit([0, horizon], [current_state[2], d_end], 5)path = {'s': np.polyval(s_coeff, t),'d': np.polyval(d_coeff, t),'s_dot': np.polyval(np.polyder(s_coeff), t),'d_dot': np.polyval(np.polyder(d_coeff), t)}paths.append(path)return paths
def select_optimal_path(paths, obstacles):min_cost = float('inf')best_path = Nonefor path in paths:# 计算舒适性成本jerk_cost = np.mean(np.diff(path['s_dot'])**2)# 计算安全成本collision_cost = 0for t in range(len(path['s'])):x = path['s'][t]y = path['d'][t]for obs in obstacles:dist = np.sqrt((x - obs['x'])**2 + (y - obs['y'])**2)if dist < (obs['r'] + 1.5):  # 1.5m为安全距离collision_cost += 1 / max(dist, 0.1)# 计算总成本total_cost = 0.4*jerk_cost + 0.3*collision_costif total_cost < min_cost:min_cost = total_costbest_path = pathreturn best_path
def plot_scenario(paths, best_path, obstacles):plt.figure(figsize=(10, 6))# 绘制障碍物for obs in obstacles:circle = plt.Circle((obs['x'], obs['y']), obs['r'], color='r', alpha=0.3)plt.gca().add_patch(circle)# 绘制所有候选路径for path in paths:plt.plot(path['s'], path['d'], 'gray', alpha=0.2)# 高亮最优路径plt.plot(best_path['s'], best_path['d'], 'b', linewidth=2)plt.xlabel('Longitudinal Position (m)')plt.ylabel('Lateral Position (m)')plt.grid(True)plt.axis('equal')plt.show()# 主程序
if __name__ == "__main__":ego_state = [20, 10, 0, 0]  # [s, s_dot, d, d_dot]candidate_paths = generate_lattice_paths(ego_state)optimal_path = select_optimal_path(candidate_paths, obstacles)plot_scenario(candidate_paths, optimal_path, obstacles)

四、运行结果分析

在这里插入图片描述

(注:实际运行后将显示包含障碍物区域和最优路径的轨迹图,灰色为候选路径,蓝色为最优避障路径)


五、工程优化建议

  1. 引入Frenet坐标系:将全局坐标转换为(s,d)坐标系,简化横向/纵向解耦计算
  2. 动态权重调整:根据场景危险程度自动调整成本函数权重(如障碍物接近时加大安全权重)
  3. 运动学约束检查:添加最大曲率、加速度边界条件校验
  4. 多线程优化:对候选轨迹的代价计算进行并行加速

六、结语

通过不到100行的Python代码,我们实现了一个完整的Lattice Planner避障规划原型系统。该代码在Python 3.8+环境下可直接运行(需安装numpy和matplotlib)。在实际自动驾驶系统中,还需融合高精地图、感知预测模块等,但核心的轨迹生成与优化思想与此文一脉相承。

技术亮点

  1. 采用多项式参数化方法保证轨迹平滑性
  2. 引入多维度加权成本函数实现最优决策
  3. 模块化设计便于功能扩展

原创声明:本文代码及算法思路均为作者原创,转载需注明出处。关注技术博客获取更多自动驾驶实战教程!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/84783.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

切换到旧提交,同时保证当前修改不丢失

在 Git 中&#xff0c;可以通过以下几种方式切换到之前的提交&#xff0c;同时保留当前的提交&#xff08;即不丢失工作进度&#xff09;&#xff1a; 1. 使用 git checkout 创建临时分离头指针&#xff08;推荐用于查看&#xff09; git checkout <commit-hash>这会让…

zookeeper 操作总结

zookeeper 中的节点类型 节点类型命令选项说明‌持久节点‌无选项&#xff08;默认&#xff09;永久存在&#xff0c;除非手动删除。‌临时节点‌-e与客户端会话绑定&#xff0c;会话结束自动删除&#xff08;‌不能有子节点‌&#xff09;。‌顺序节点‌-s节点名自动追加递增…

nova14 ultra,是如何防住80°C热水和10000KPa水压冲击的?

暴雨突袭&#xff0c;手忙脚乱护住背包&#xff0c;却担心手机被雨水浸湿&#xff1b;泳池里想记录美好时刻&#xff0c;却担心手机掉入水中 &#xff1b;厨房里充满了高温水汽&#xff0c;近距离拍摄美食瞬间&#xff0c;手机屏幕花屏&#xff0c;让人失去了对美食的兴趣…… …

flutter加载dll 报错问题

解决flutter加载dll 报错问题 LoadLibrary 报错 126 or 193 明确一点&#xff1a;flutter构建exe 时默认是MSVC的。 1. 先检查dll 的位数是否满足 file ***.dll output: PE32 executable (DLL) (console) x86-64, for MS Windows, 19 sections 这种是64位的机器。 满足的话可…

Mac 版不能连接华为 GaussDB 吗?我看 Windows 版可以连接?

&#x1f9d1;‍&#x1f4bb; GaussDB 用户 Mac 版不能连接华为 GaussDB 吗&#xff1f;我看Windows 版可以连接。 &#x1f9d1;‍&#x1f527; 官方技术中心 由于 GaussDB 数据库本身未支持 macOS 系统&#xff0c;所以在 macOS 上的 Navicat 中也未支持该数据库。 &…

【MySQL成神之路】MySQL索引相关介绍

1 相关理论介绍 一、索引基础概念 二、索引类型 1. 按数据结构分类 2. 按功能分类 三、索引数据结构原理 B树索引特点&#xff1a; 哈希索引特点&#xff1a; 四、索引使用原则 1. 创建索引原则 2. 避免索引失效情况 五、索引优化策略 六、索引维护与管理 七、特殊…

五、web安全--XSS漏洞(1)--XSS漏洞利用全过程

本文章仅供学习交流&#xff0c;如作他用所承受的法律责任一概与作者无关1、XSS漏洞利用全过程 1.1 寻找注入点&#xff1a;攻击者首先需要找到目标网站中可能存在XSS漏洞的注入点。这些注入点通常出现在用户输入能够直接输出到页面&#xff0c;且没有经过适当过滤或编码的地方…

使用 Shell 脚本实现 Spring Boot 项目自动化部署到 Docker(Ubuntu 服务器)

使用 Shell 脚本实现 Spring Boot 项目自动化部署到 Docker&#xff08;Ubuntu 服务器&#xff09; 在日常项目开发中&#xff0c;我们经常会将 Spring Boot 项目打包并部署到服务器上的 Docker 环境中。为了提升效率、减少重复操作&#xff0c;我们可以通过 Shell 脚本实现自动…

高考加油(Python+HTML)

前言 询问DeepSeek根据自己所学到的知识来生成多个可执行的代码&#xff0c;为高考学子加油。最开始生成的都会有点小问题&#xff0c;还是需要自己调试一遍&#xff0c;下面就是完整的代码&#xff0c;当然了最后几天也不会有多少人看&#xff0c;都在专心的备考。 Python励…

HTTP协议接口三种测试方法之-JMeter(保姆教程)

在当今 API 驱动的开发世界中&#xff0c;高效、可靠的 HTTP 接口测试是保障应用质量的关键。作为开源性能测试工具中的王者&#xff0c;Apache JMeter 不仅擅长压力测试&#xff0c;更是进行功能性和回归测试的利器。本文将手把手教你如何用 JMeter 构建强大的 HTTP 测试计划&…

聊聊JVM怎么调优?(实战总结)

JVM 核心配置与调优指南 一、堆内存与年轻代配置&#xff08;影响最大&#xff09; 堆内存大小&#xff1a; 在资源允许的前提下&#xff0c;堆内存应尽可能设置得更大。关键点&#xff1a; 必须将堆内存的最大值 (-Xmx) 和最小值 (-Xms) 设置为相同值。动态扩容会触发 Full G…

开疆智能Profinet转Profibus网关连接费斯托阀岛总线模块配置案例

本案例是通过开疆智能Profibus转Profinet网关将费托斯阀岛接入到西门子1200PLC的配置案例。 首先我们先了解一下Profibus报文以及他的通讯原理。 除了起始符 SD 和结束符 ED 这些固定数值之外&#xff0c;还有功能码&#xff08;Function Code, FC&#xff09;和服务访问点&…

ARM内核一览

经常看介绍某某牛批芯片用的又是ARM什么核&#xff0c;看的云里雾里&#xff0c;所以简单整理整理。&#xff08;内容来自官网和GPT&#xff09; 1 ARM 内核总体分类 系列特点应用场景Cortex-M超低功耗、低成本、实时性嵌入式系统、微控制器、IoTCortex-R高可靠性、硬实时汽车…

RT Thread Nano V4.1.1 rtconfig.h 注释 Configuration Wizard 格式

rtcomfig.h 以下是对 [rtconfig.h](file://c:\Users\admin\Downloads\rtthread-nano-master\rt-thread\bsp\stm32f407-msh\RT-Thread\rtconfig.h) 文件中每一个配置项的详细注释说明: 基本配置(Basic Configuration) [RT_THREAD_PRIORITY_MAX](file://c:\Users\admin\Downl…

UniApp网页版集成海康视频播放器

注意&#xff1a;本人全部集成好后使用最新的海康平台下载插件进行替换后就不能预览视频 使用Uni插件进行集成&#xff1a;海康视频H5播放器组件 - DCloud 插件市场 CSDN资源下载&#xff1a;https://download.csdn.net/download/wangdaoyin2010/90910975 注意&#xff1a;初…

WPF【10_2】数据库与WPF实战-示例

客户预约关联示例图 MainWindow.xaml 代码 <Window x:Class"WPF_CMS.MainWindow" xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d"ht…

理解 Kubernetes 的架构与控制平面组件运行机制

文章目录 K8s架构K8s核心组件控制平面组件&#xff08;部署在 Master 节点&#xff09;1. 查看组件运行情况2. 查看组件 help 命令 Node端组件&#xff08;部署在每个工作节点&#xff09; K8s内部工作原理 Kubernetes&#xff08;也称为 K8s&#xff09;是一个开源的容器编排和…

Express+MySQL后台开发实战:从模块化到错误处理的全链路解析

ExpressMySQL后台开发实战&#xff1a;从模块化到错误处理的全链路解析 摘要&#xff1a;本文将以Node.jsExpress框架为基础&#xff0c;结合MySQL数据库实战&#xff0c;深度剖析后台系统中数据库模块化设计、安全查询、错误处理等核心开发要点。 一、项目环境与技术栈 ├─…

Spring AI 智能体代理模式(Agent Agentic Patterns)

AgentAgenticPatterns 简介 在最近的一篇研究报告《构建高效代理》 中&#xff0c;Anthropic分享了关于构建高效大语言模型&#xff08;LLM&#xff09;代理的宝贵见解。这项研究特别有趣的地方在于&#xff0c;它强调简单性和可组合性&#xff0c;而非复杂的框架。让我们来探…

基于 Vue3 与 exceljs 实现自定义导出 Excel 模板

在开发中&#xff0c;我们需要常常为用户提供更多的数据录入方式&#xff0c;Excel 模板导出与导入是一个常见的功能点。本文将介绍如何使用 Vue3、exceljs 和 file-saver 实现一个自定义导出 Excel 模板&#xff0c;并在特定列添加下拉框选择的数据验证功能。 技术选型 excelj…