PID控制在自动驾驶中的应用举例(二)航向控制

2023-11-09 01:59

本文主要是介绍PID控制在自动驾驶中的应用举例(二)航向控制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一篇中介绍了使用PID进行车速控制,控制目标相对简单,如果加入转向的目标,任务复杂程度都会有所增加。

对于环境的配置与之前类似,不再赘述。

from matplotlib import pyplot as plt
from collections import deque
import numpy as npimport gym
import highway_env
%matplotlib inlineenv = gym.make('highway-v0')
config = \{"observation": {"type": "Kinematics","vehicles_count": 1,"features": ["presence",'x','y', "vx", "vy"],"features_range": {"x": [-100, 100],"y": [-100, 100],"vx": [-100, 100],"vy": [-100, 100]},"absolute": True,"order": "sorted"},"action": {"type": "ContinuousAction"},    "simulation_frequency": 15,  # [Hz]"policy_frequency": 5,  # [Hz]'vehicles_count': 0,'reward_speed_range': [20, 80],}

对于航向控制来说,我自己总结了一种计算方向盘转角的方式,需要输入3个矢量:汽车朝向,目的地要求的汽车朝向和汽车位置与目的地连线的朝向。目的是根据当前汽车的朝向和相对于目的地的方向推算出方向盘合适的转角度数。
方位示意图
将车辆和目的地的方向矢量相加,再与direction的矢量角度进行比较,即可确定方向盘需旋转的角度,如果δ为负(逆时针),则向左打方向盘,若为正(顺时针),向右打方向盘。角度的大小由向量夹角公式决定。
在这里插入图片描述

使用代码对以上方法进行表示:

def get_angle(v_p,v_h,dest_p,dest_h):v_head_x=np.cos(v_h)v_head_y=np.sin(v_h)  d_head_x=1*np.cos(dest_h)d_head_y=1*np.sin(dest_h)car_dir = np.array([v_head_x,v_head_y])path_dir = np.array([dest_p[0]-v_p[0],dest_p[1]-v_p[1]])dest_dir = np.array([car_dir[0]+d_head_x,car_dir[1]+d_head_y])cos_theta=np.dot(dest_dir,path_dir)/(np.linalg.norm(dest_dir)*np.linalg.norm(path_dir))left_right = 0 if np.cross(dest_dir,path_dir)==0 else abs(np.cross(dest_dir,path_dir))/np.cross(dest_dir,path_dir)angle = np.arccos(cos_theta)*left_rightreturn angle

之后可以手动指定车辆和目的地的坐标和方向,坐标是我随机设的点,对于highway-env环境来说,车道不是一个严格的物理模型,汽车开出车道也不会停止模拟。方向的范围是[0,2*pi]。

V_HEADING=np.pi
V_POSITION_X=180
V_POSITION_Y=2D_HEADING=0
D_POSITION_X=360
D_POSITION_Y=12

之后就可以使用PID等控制方法对汽车进行导航。为了在到达目的地之后停止环境模拟,可以设置一个距离函数,当汽车和目的地的距离小于2m时,认为汽车已经到达目的地,停止模拟。

def get_distance(v_p,dest_p):dis=np.sqrt((v_p[0]-dest_p[0])**2+(v_p[1]-dest_p[1])**2)return dis

传统控制

传统控制直接将δ值作为方向盘转角steering值。由于物理环境没有阻力,不用对油门踏板进行设置,汽车会一直按照初始速度匀速前进。

env.configure(config)
env.reset()
env.vehicle.heading=V_HEADING
env.vehicle.position=[V_POSITION_X,V_POSITION_Y]
dest_position=[D_POSITION_X,D_POSITION_Y]
dest_heading=D_HEADING
e=0
his_p1=[]for _ in range(300):action=[0,e]obs, reward, done, info = env.step(action)v_p=env.road.vehicles[0].positionv_h=env.vehicle.headinghis_p1.append([v_p[0],v_p[1]])angle=get_angle(v_p,v_h,dest_position,dest_heading)e = 0 if abs(angle)<0.01 else min(max(angle,-1),1)  # [-3.14,3.14] ->  [-1,1]dis = get_distance(v_p,dest_position)if dis<2:breakenv.render()
env.close()

PID控制

与速度控制类似,PID需要开辟一个buffer以便求微分和积分。

env.configure(config)
env.reset()
env.vehicle.heading=V_HEADING
env.vehicle.position=[V_POSITION_X,V_POSITION_Y]
dest_position=[D_POSITION_X,D_POSITION_Y]
dest_heading=D_HEADING
dt=0.1
buffer = deque(maxlen=10)e=0
e_p=0
e_i=0
e_d=0his_p2=[]
for _ in range(300):action=[0,e]obs, reward, done, info = env.step(action)v_p=env.road.vehicles[0].positionv_h=env.vehicle.headinghis_p2.append([v_p[0],v_p[1]])angle=get_angle(v_p,v_h,dest_position,dest_heading)    e_p = 0 if abs(angle)<0.01 else min(max(angle,-1),1)buffer.append(e_p)e_i=np.sum(buffer)*dtif len(buffer)>=2:        e_d=(buffer[-1]-buffer[-2])/dtelse:e_d=0e=e_p+0.5*e_i+0.05*e_ddis = get_distance(v_p,dest_position)if dis<2:breakenv.render()
env.close()

模拟完成后可以画出汽车行驶路径的散点图,以观察两种方法的异同

plt.figure(figsize=(12,8))
plt.scatter(np.transpose(his_p1)[0],-np.transpose(his_p1)[1])
plt.scatter(np.transpose(his_p2)[0],-np.transpose(his_p2)[1])

在这里插入图片描述
(地图里y轴正负是反的,我也不理解为啥画出来的图y轴值都是负的)

可以看出来方向控制的变化和速度控制类似,采用PID可以快速将方向调整到的和目标方向一致,但会有小幅震荡。

这篇关于PID控制在自动驾驶中的应用举例(二)航向控制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/373541

相关文章

Spring AI集成DeepSeek三步搞定Java智能应用的详细过程

《SpringAI集成DeepSeek三步搞定Java智能应用的详细过程》本文介绍了如何使用SpringAI集成DeepSeek,一个国内顶尖的多模态大模型,SpringAI提供了一套统一的接口,简... 目录DeepSeek 介绍Spring AI 是什么?Spring AI 的主要功能包括1、环境准备2

Python中如何控制小数点精度与对齐方式

《Python中如何控制小数点精度与对齐方式》在Python编程中,数据输出格式化是一个常见的需求,尤其是在涉及到小数点精度和对齐方式时,下面小编就来为大家介绍一下如何在Python中实现这些功能吧... 目录一、控制小数点精度1. 使用 round() 函数2. 使用字符串格式化二、控制对齐方式1. 使用

Spring AI与DeepSeek实战一之快速打造智能对话应用

《SpringAI与DeepSeek实战一之快速打造智能对话应用》本文详细介绍了如何通过SpringAI框架集成DeepSeek大模型,实现普通对话和流式对话功能,步骤包括申请API-KEY、项目搭... 目录一、概述二、申请DeepSeek的API-KEY三、项目搭建3.1. 开发环境要求3.2. mav

Springboot的自动配置是什么及注意事项

《Springboot的自动配置是什么及注意事项》SpringBoot的自动配置(Auto-configuration)是指框架根据项目的依赖和应用程序的环境自动配置Spring应用上下文中的Bean... 目录核心概念:自动配置的关键特点:自动配置工作原理:示例:需要注意的点1.默认配置可能不适合所有场景

Springboot控制反转与Bean对象的方法

《Springboot控制反转与Bean对象的方法》文章介绍了SpringBoot中的控制反转(IoC)概念,描述了IoC容器如何管理Bean的生命周期和依赖关系,它详细讲解了Bean的注册过程,包括... 目录1 控制反转1.1 什么是控制反转1.2 SpringBoot中的控制反转2 Ioc容器对Bea

MobaXterm远程登录工具功能与应用小结

《MobaXterm远程登录工具功能与应用小结》MobaXterm是一款功能强大的远程终端软件,主要支持SSH登录,拥有多种远程协议,实现跨平台访问,它包括多会话管理、本地命令行执行、图形化界面集成和... 目录1. 远程终端软件概述1.1 远程终端软件的定义与用途1.2 远程终端软件的关键特性2. 支持的

Java中实现订单超时自动取消功能(最新推荐)

《Java中实现订单超时自动取消功能(最新推荐)》本文介绍了Java中实现订单超时自动取消功能的几种方法,包括定时任务、JDK延迟队列、Redis过期监听、Redisson分布式延迟队列、Rocket... 目录1、定时任务2、JDK延迟队列 DelayQueue(1)定义实现Delayed接口的实体类 (

shell脚本自动删除30天以前的文件(最新推荐)

《shell脚本自动删除30天以前的文件(最新推荐)》该文章介绍了如何使用Shell脚本自动删除指定目录下30天以前的文件,并通过crontab设置定时任务,此外,还提供了如何使用Shell脚本删除E... 目录shell脚本自动删除30天以前的文件linux按照日期定时删除elasticsearch索引s

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

Java中八大包装类举例详解(通俗易懂)

《Java中八大包装类举例详解(通俗易懂)》:本文主要介绍Java中的包装类,包括它们的作用、特点、用途以及如何进行装箱和拆箱,包装类还提供了许多实用方法,如转换、获取基本类型值、比较和类型检测,... 目录一、包装类(Wrapper Class)1、简要介绍2、包装类特点3、包装类用途二、装箱和拆箱1、装