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

相关文章

一文详解MySQL如何设置自动备份任务

《一文详解MySQL如何设置自动备份任务》设置自动备份任务可以确保你的数据库定期备份,防止数据丢失,下面我们就来详细介绍一下如何使用Bash脚本和Cron任务在Linux系统上设置MySQL数据库的自... 目录1. 编写备份脚本1.1 创建并编辑备份脚本1.2 给予脚本执行权限2. 设置 Cron 任务2

Python包管理工具核心指令uvx举例详细解析

《Python包管理工具核心指令uvx举例详细解析》:本文主要介绍Python包管理工具核心指令uvx的相关资料,uvx是uv工具链中用于临时运行Python命令行工具的高效执行器,依托Rust实... 目录一、uvx 的定位与核心功能二、uvx 的典型应用场景三、uvx 与传统工具对比四、uvx 的技术实

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

MyBatis Plus 中 update_time 字段自动填充失效的原因分析及解决方案(最新整理)

《MyBatisPlus中update_time字段自动填充失效的原因分析及解决方案(最新整理)》在使用MyBatisPlus时,通常我们会在数据库表中设置create_time和update... 目录前言一、问题现象二、原因分析三、总结:常见原因与解决方法对照表四、推荐写法前言在使用 MyBATis

Python使用smtplib库开发一个邮件自动发送工具

《Python使用smtplib库开发一个邮件自动发送工具》在现代软件开发中,自动化邮件发送是一个非常实用的功能,无论是系统通知、营销邮件、还是日常工作报告,Python的smtplib库都能帮助我们... 目录代码实现与知识点解析1. 导入必要的库2. 配置邮件服务器参数3. 创建邮件发送类4. 实现邮件

Java中常见队列举例详解(非线程安全)

《Java中常见队列举例详解(非线程安全)》队列用于模拟队列这种数据结构,队列通常是指先进先出的容器,:本文主要介绍Java中常见队列(非线程安全)的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一.队列定义 二.常见接口 三.常见实现类3.1 ArrayDeque3.1.1 实现原理3.1.2

Python远程控制MySQL的完整指南

《Python远程控制MySQL的完整指南》MySQL是最流行的关系型数据库之一,Python通过多种方式可以与MySQL进行交互,下面小编就为大家详细介绍一下Python操作MySQL的常用方法和最... 目录1. 准备工作2. 连接mysql数据库使用mysql-connector使用PyMySQL3.

如何搭建并配置HTTPD文件服务及访问权限控制

《如何搭建并配置HTTPD文件服务及访问权限控制》:本文主要介绍如何搭建并配置HTTPD文件服务及访问权限控制的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、安装HTTPD服务二、HTTPD服务目录结构三、配置修改四、服务启动五、基于用户访问权限控制六、

Python使用Tkinter打造一个完整的桌面应用

《Python使用Tkinter打造一个完整的桌面应用》在Python生态中,Tkinter就像一把瑞士军刀,它没有花哨的特效,却能快速搭建出实用的图形界面,作为Python自带的标准库,无需安装即可... 目录一、界面搭建:像搭积木一样组合控件二、菜单系统:给应用装上“控制中枢”三、事件驱动:让界面“活”

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹