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

相关文章

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

SpringBoot项目启动后自动加载系统配置的多种实现方式

《SpringBoot项目启动后自动加载系统配置的多种实现方式》:本文主要介绍SpringBoot项目启动后自动加载系统配置的多种实现方式,并通过代码示例讲解的非常详细,对大家的学习或工作有一定的... 目录1. 使用 CommandLineRunner实现方式:2. 使用 ApplicationRunne

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

oracle中exists和not exists用法举例详解

《oracle中exists和notexists用法举例详解》:本文主要介绍oracle中exists和notexists用法的相关资料,EXISTS用于检测子查询是否返回任何行,而NOTE... 目录基本概念:举例语法pub_name总结 exists (sql 返回结果集为真)not exists (s

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

python实现自动登录12306自动抢票功能

《python实现自动登录12306自动抢票功能》随着互联网技术的发展,越来越多的人选择通过网络平台购票,特别是在中国,12306作为官方火车票预订平台,承担了巨大的访问量,对于热门线路或者节假日出行... 目录一、遇到的问题?二、改进三、进阶–展望总结一、遇到的问题?1.url-正确的表头:就是首先ur