【自动驾驶】控制算法(八)横向控制Ⅲ | 代码与模型

news2024/11/16 21:23:41

写在前面:
🌟 欢迎光临 清流君 的博客小天地,这里是我分享技术与心得的温馨角落。📝
个人主页:清流君_CSDN博客,期待与您一同探索 移动机器人 领域的无限可能。

🔍 本文系 清流君 原创之作,荣幸在CSDN首发🐒
若您觉得内容有价值,还请评论告知一声,以便更多人受益。
转载请注明出处,尊重原创,从我做起。

👍 点赞、评论、收藏,三连走一波,让我们一起养成好习惯😜
在这里,您将收获的不只是技术干货,还有思维的火花

📚 系列专栏:【运动控制】系列,带您深入浅出,领略控制之美。🖊
愿我的分享能为您带来启迪,如有不足,敬请指正,让我们共同学习,交流进步!

🎭 人生如戏,我们并非能选择舞台和剧本,但我们可以选择如何演绎 🌟
感谢您的支持与关注,让我们一起在知识的海洋中砥砺前行~~~


文章目录

  • 引言
  • 一、控制算法编程基础
  • 二、Carsim配置
    • 1、整车参数获取与侧偏刚度计算
    • 2、车辆状态与规划轨迹信息
  • 三、路径规划与轨迹生成
    • 1、路径规划代码概述
    • 2、直线与圆弧子函数详解
      • (1)直线函数使用方法
      • (2)圆弧函数使用方法
    • 3、路径拼接实例
    • 4、Carsim路面设置
  • 四、模型搭建与仿真运行
    • 1、变量准备
    • 2、预测模块实现
    • 3、误差和曲率计算模块实现
    • 4、AB模块 + LQR模块
    • 5、前馈模块实现
    • 6、避免代数环的延迟模块
    • 7、仿真测试
  • 五、总结
  • 参考资料


引言

  本篇博客是 自动驾驶控制算法 系列的第八节第Ⅲ部分。内容整理自 B站知名up主 忠厚老实的老王 的视频,作为博主的学习笔记,分享给大家共同学习。

  本篇博客是最长的一节,主要讲解如何搭建模型。

  大家有需要可以自己下载相关的代码模型,主页链接如下:

  自动驾驶决策规划算法代码模型


一、控制算法编程基础

  在开始本节之前,首先要把第八节的第二部分 Carsim 设置给设置完毕。具体操作见第八节第Ⅱ部分:

  【自动驾驶】控制算法(八)横向控制Ⅱ | Carsim 与 Matlab 联合仿真基本操作

  在开始本节之前,首先要有算法准备,就是第八讲第一节的编程基础内容:

  【自动驾驶】控制算法(八)横向控制Ⅰ | 算法与流程

  编程就是照着算法流程图编写程序:
在这里插入图片描述
这是整个编程的核心,横向控制算法主要需要三类参数:

  • 整车参数
  • 车辆当前状态
  • 规划轨迹信息

二、Carsim配置

1、整车参数获取与侧偏刚度计算

  整车参数可以在Carsim软件里读取,可得到质量 m m m、质心到前后轮的距离 a a a b b b 以及惯量 I I I 之类的数据。

  注意:质量只是弹簧上质量,不包括悬架质量,所以

车辆总质量 = 簧上质量 + 前后两个悬架质量 车辆总质量=簧上质量+前后两个悬架质量 车辆总质量=簧上质量+前后两个悬架质量  其他参数在软件里可以读取到,包括根据曲线估算的侧偏刚度。侧偏刚度可以不用计算特别准,只要估算出大概值即可。

  计算侧偏刚度,先大概估计四轮垂向力的大小,在表上查垂向力对应的曲线,求斜率。

  注意:侧偏刚度的单位是 N / r a d N/rad N/rad,且为负值。

  当输入时,算法是牛每弧度,要进行单位换算,并且所有侧偏刚度都要乘以 2 2 2,因为是自行车模型,把两个轮胎并成一个轮胎。在这里算出来的侧偏刚度是正的,但是当输入进软件里时,侧偏刚度是负的。

  注意:把风阻、空气动力学关掉。

在这里插入图片描述

  暂时不考虑风阻,先简单再复杂,把空气动力学关掉。

  其实风阻在控制中是很要命的因素,特别是在高速时,空气阻力会极大影响车轮的垂向力,垂向力变化,侧偏刚度就变了,导致 LQR 相关东西全都变了,所以风阻是很重要的因素,但如果讲风阻就太复杂了,先暂时先不考虑风阻,等以后学到时再去考虑风阻。

2、车辆状态与规划轨迹信息

  回到流程图,除了整车参数,算法还需要车辆状态,也就是 ( x , y , φ , v x , v y , φ ˙ ) (x,y,\varphi ,v_x,v_y,\dot{\varphi}) (x,y,φ,vx,vy,φ˙)。所以 Carsim 的输出中要添加 ( x , y , φ , v x , v y , φ ˙ ) (x,y,\varphi ,v_x,v_y,\dot{\varphi}) (x,y,φ,vx,vy,φ˙)
在这里插入图片描述

  流程图的第三个输入是规划轨迹信息,包括 ( x r , y r , θ r , κ r ) \left( x_r,y_r,\theta _r,\kappa _r \right) (xr,yr,θr,κr),规划轨迹的横纵坐标、规划轨迹的切线与 x x x 轴方向夹角、规划曲线的曲率。

  所以做控制时不仅仅写控制算法,还要写规划算法,先把模型的框架搭建立起来,点Send to Simulink


三、路径规划与轨迹生成

1、路径规划代码概述

  规划代码不是这一节的重点,所以规划代码已经事先写好了,看起来挺复杂,实际上就写了两个子函数,在设计路径时,调用这两个子函数把拼起来就可以了。规划代码如下:

routing_planning.m

count=50;
[x1,y1,theta1,kr1]=straight([0,0],[20,0],0,count);
[x2,y2,theta2,kr2]=arc([20,0],[30,10],0,pi/2,count);
[x3,y3,theta3,kr3]=arc([30,10],[40,20],pi/2,0,count);
[x4,y4,theta4,kr4]=arc([40,20],[40,40],0,pi,count);
[x5,y5,theta5,kr5]=arc([40,40],[35,35],pi,3*pi/2,count);
[x6,y6,theta6,kr6]=arc([35,35],[25,35],3*pi/2,pi/2,count);
[x7,y7,theta7,kr7]=arc([25,35],[15,35],pi/2,3*pi/2,count);
[x8,y8,theta8,kr8]=arc([15,35],[5,35],3*pi/2,pi/2,count);
[x9,y9,theta9,kr9]=arc([5,35],[-15,35],pi/2,3*pi/2,count);
[x10,y10,theta10,kr10]=straight([-15,35],[-15,15],3*pi/2,count);
[x11,y11,theta11,kr11]=arc([-15,15],[0,0],3*pi/2,2*pi,count);
xr=[x1,x2,x3,x4,x5,x6,x7,x8,x9,x10,x11];
yr=[y1,y2,y3,y4,y5,y6,y7,y8,y9,y10,y11];
thetar=[theta1,theta2,theta3,theta4,theta5,theta6,theta7,theta8,theta9,theta10,theta11];
kappar=[kr1,kr2,kr3,kr4,kr5,kr6,kr7,kr8,kr9,kr10,kr11];

scatter(xr,yr);

function[xr,yr,thetar,kr]=straight(init_coord,end_coord,init_angle,count)
delta_x=(end_coord(1)-init_coord(1))/(count-1);
delta_y=(end_coord(2)-init_coord(2))/(count-1);
for i=1:count
    xr(i)=init_coord(1)+delta_x*i;
    yr(i)=init_coord(2)+delta_y*i;
    thetar(i)=init_angle;
    kr(i)=0;
end      
end

function[xr,yr,thetar,kr]=arc(init_coord,end_coord,init_angle,end_angle,count)
    L=sqrt((init_coord(1)-end_coord(1))^2+(init_coord(2)-end_coord(2))^2);
    R=L/sqrt(2*(1-cos(end_angle-init_angle)));
    delta_angle=(end_angle-init_angle)/(count-1) ;
  
       for i=1:count
           if delta_angle>0
               xr(i)=init_coord(1)-R*sin(init_angle)+R*sin(init_angle+delta_angle*(i-1));
               yr(i)=init_coord(2)+R*cos(init_angle)-R*cos(init_angle+delta_angle*(i-1));
               thetar(i)=init_angle+delta_angle*i;
               kr(i)=1/R;
           else
               xr(i)=init_coord(1)+R*sin(init_angle)-R*sin(init_angle+delta_angle*(i-1));

               yr(i)=init_coord(2)-R*cos(init_angle)+R*cos(init_angle+delta_angle*(i-1));
               thetar(i)=init_angle+delta_angle*i;
               kr(i)=-1/R;
           end               
       end  
end

2、直线与圆弧子函数详解

  具体讲一下这两个子函数该怎么用,这两个子函数的名字是直线、圆弧。代码上半部分的内容是拼起来的路径,也就是要用的路径,暂时先注释掉。

(1)直线函数使用方法

  这两个子函数用起来很简单,比如想生成一段直线,起点是 ( 0 , 0 ) (0,0) (0,0),终点是 ( 5 , 0 ) (5,0) (5,0),角度是 π 4 \frac{\pi}{4} 4π,只需要直线函数就可以:

[xr,yr,~,~] = straight([0,0],[50,50],pi/4,10);

  严格来说要写 θ r , κ r \theta_r,\kappa_r θr,κr,不过不需要的话就用 ~ 占位,生成这样一条离散路径点:
在这里插入图片描述

(2)圆弧函数使用方法

  圆弧稍微复杂一点,需要输入起点坐标、终点坐标、起点角度、终点角度、点的数量。起点角度,终点角度表示点的切线方向与 x x x 轴的夹角。

[xr,yr,~,~] = arc([0,0],[50,50],0,pi/2,100);

  比如这里起点角度是 0 ° 0° ,终点角度是 π 2 \frac{\pi}{2} 2π,运行结果如下:
在这里插入图片描述

  此圆弧是典型左转圆弧,可以修改参数,比如把 π 2 \frac{\pi}{2} 2π 改成 − π 2 -\frac{\pi}{2} 2π,同样终点坐标要从 ( 50 , 50 ) (50,50) (50,50) 变成 ( 50 , − 50 ) (50,- 50) (50,50),这样就变成右转圆弧。还可以修改点数量。

  这两个子函数的作用就是给路径形状,能把 x r , y r , θ r , κ r x_r,y_r,\theta_r,\kappa_r xr,yr,θr,κr 直接算出来,很方便,在设计路径时,只要调用这两个接口,拼起来就可以了。可以自己设计这样的路径出来。

3、路径拼接实例

  本篇博客所演示的路径比较复杂,各种直线和圆弧拼在一起,具体路径如下图所示:
在这里插入图片描述

  每一段直线和圆弧都由 50 50 50 个点组成,比较复杂。这里的路径规划其实设计的很不合理,因为好的路径规划要保证曲率连续,但是本路径没有做到,就是简单的直线和圆弧的拼接,直线的曲率是 0 0 0,圆弧的曲率是 1 / 2 1/ 2 1/2,也就是曲率半径的倒数。

  要设计比较好的路径,需要在直线和圆弧中间用回旋线做过渡,让曲率连续变化。但是因为规划并不是本篇博客的重点,所以就凑合用一用, count 是控制点的数量,选择点多一点的,这样控制效果相对会好一点。

4、Carsim路面设置

  为了看起来更直观,把 Carsim 的路面也设置一下。

  点Procedure,在 3D Road里面选 1200 1200 1200 米的道路:

在这里插入图片描述

  在车道这里选 two lines,再点 stretch east,在这里根据规划设计道路走向。
在这里插入图片描述

  上图已经设置好。如果无法修改,先把右上角的锁点开,这样就可以修改了。如果没有 11 11 11 行这样的表,点右下角有Rows输入 11 11 11,就会有 11 11 11 行表格。表格的参数按上图填写即可, straight 代表直线, radius 代表圆弧。

  圆弧转过的角度,即左转右转和规划的符号不一样,左转默认半径为正,右转默认半径为负。在规划代码里,若终点角度如果大于起点角度,就是左转;反之就是右转。在这里是根据半径来的,半径为负,就是右转;半径是正,就是左转。一定要把 Treat as loop 勾上,变成闭合曲线处理。

  设计的道路挺扭曲,转弯半径非常小,最小的只有 5 m 5m 5m,而且在上面有连续五米的转弯半径,实际上对车辆的通过能力是很大的考验,因为转弯半径特别小,如果不是倒车,转弯半径一般最低也就 8 m 8m 8m

  而 5 m 5m 5m 的转弯半径已经特别陡了,所以设计的路径比较严苛,创建的道路如下图所示:
在这里插入图片描述

  有很多连续弯道,路况算比较严苛。


四、模型搭建与仿真运行

1、变量准备

  先跑一遍路径规划,就是Matlab变量工作区里要有路径才可以:

在这里插入图片描述

  打开 Simulink 看一下控制模型,先做单位换算,把千米每小时换成米每秒,以及把角度换成弧度。建立 Matlab Function 模块,在里面编程。

  这六个模块从逻辑顺序来上来说,从左往右写比较好,因为左边的输出是右边的输入,所以先写左边,再写右边。因此先写预测模块,在写代码之前先写上名字,因为模型比较复杂,先标上记号,要不然可能不知道哪根线对应哪个变量。

2、预测模块实现

  预测模块的输入是当前点的 ( x , y , φ , v x , v y , φ ˙ ) (x,y,\varphi ,v_x,v_y,\dot{\varphi}) (x,y,φ,vx,vy,φ˙),输出为预测后的 ( x p r e , y p r e , φ p r e , v x p r e , v y p r e , φ ˙ p r e ) (x_{pre},y_{pre},\varphi_{pre} ,v_{xpre},v_{ypre},\dot{\varphi}_{pre}) (xpre,ypre,φpre,vxpre,vypre,φ˙pre)。照着预测模块的算法写,预测时间随便给个 0.1 0.1 0.1 秒。

predict_module

function [pre_x,pre_y,pre_phi,pre_vx,pre_vy,pre_phi_dot] = fcn(x,y,phi,vx,vy,phi_dot,ts)
    pre_x=x+vx*ts*cos(phi)-vy*ts*sin(phi);
    pre_y=y+vy*ts*cos(phi)+vx*ts*sin(phi);
    pre_phi=phi+phi_dot*ts;
    pre_vx=vx;
    pre_vy=vy;
    pre_phi_dot=phi_dot;
end

3、误差和曲率计算模块实现

  计算模块需要车辆状态,还需要规划轨迹信息。所以规划要先把它写好,而且写好之后也必须跑一遍,规划代码和控制代码是独立的。所以在规划时,需要先把代码跑一下,在工作区里要有这些规划的变量。

  在工作区里有这些变量后,才能传到 Simlink 模型里,

  注意 ( x r , y r , θ r , κ r ) \left( x_r,y_r,\theta _r,\kappa _r \right) (xr,yr,θr,κr) 必须是行向量,不能是列向量。如果是列向量, Simlink 不识别,必须写成行向量形式。

  下面可以开始写误差和曲率的计算模块,输入是 ( x , y , φ , v x , v y , φ ˙ ) (x,y,\varphi ,v_x,v_y,\dot{\varphi}) (x,y,φ,vx,vy,φ˙),以及规划的 ( x r , y r , θ r , κ r ) \left( x_r,y_r,\theta _r,\kappa _r \right) (xr,yr,θr,κr) ,输出是同一点的曲率以及误差。具体代码如下:

err_kappa_calculate_module

function [kr,err] = fcn(x,y,phi,vx,vy,phi_dot,xr,yr,thetar,kappar)
    n=length(xr);%先找到规划轨迹的长度,共有多少个规划点
    d_min=(x-xr(1))^2+(y-yr(1))^2;%把最短距离设为真实位置与第1个点。可以不用开平方根,减少计算量,因为并不关心距离是多少,只想找到最短点的序号
    min=1;%先把最短点的序号设为1
    for i=1:n%遍历所有的点
        d=(x-xr(i))^2+(y-yr(i))^2;%计算它与真实的位置的之间的距离
        if d<d_min%如果能找到比所设置的最短距离还要短
            d_min=d;%将值赋为最短距离
            min=i;%把序点的序号赋给min
        end
    end
    %遍历一遍后,一定能找到最短的距离规划点的序号,其实就是选择排序法
    dmin=min;
    tor=[cos(thetar(dmin));sin(thetar(dmin))];
    nor=[-sin(thetar(dmin));cos(thetar(dmin))];
    d_err=[x-xr(dmin);y-yr(dmin)];
    ed=nor'*d_err;
    es=tor'*d_err;
    %projection_point_thetar=thetar(dmin);%apollo
    projection_point_thetar=thetar(dmin)+kappar(dmin)*es;%投影点的航向角
    ed_dot=vy*cos(phi-projection_point_thetar)+vx*sin(phi-projection_point_thetar);
    %%%%%%%%%
    ephi=sin(phi-projection_point_thetar);%因为角度具有多值性,使用sin来消除因为多2pi或者少2pi而导致的算法出错
    %%%%%%%%%
    s_dot=vx*cos(phi-projection_point_thetar)-vy*sin(phi-projection_point_thetar);
    s_dot=s_dot/(1-kappar(dmin)*ed);
    ephi_dot=phi_dot-kappar(dmin)*s_dot;
    kr=kappar(dmin);
    err=[ed;ed_dot;ephi;ephi_dot];
end

  代码写好后拼起来即可,其中规划轨迹点 ( x r , y r , θ r , κ r ) \left( x_r,y_r,\theta _r,\kappa _r \right) (xr,yr,θr,κr) ,用From Workspace 模块,从工作区里把变量导入到 Simlink 里必须要工作区里有 ( x r , y r , θ r , κ r ) \left( x_r,y_r,\theta _r,\kappa _r \right) (xr,yr,θr,κr) 才能导。在运行 Simlink 模型时,必须要先把规划代码运行一遍,在工作区里面有这些变量,才能导入,不然会出错。

  注意From workspace 只能导行向量,能导列向量、不能导矩阵,也不能导结构体。实际上通过技巧操作可以导矩阵以及列向量,但省事点尽量变成行向量就可以了。

4、AB模块 + LQR模块

  本来 LQR 模块应该最难写,也最复杂,但是很幸运 Matlab 自己有 LQR 模块包。用离线LQR,所以 LQR 变成了最好写的模块,而且 LQR 模块和 AB 计算模块是紧密联系的,所以把 AB 计算模块和 LQR 模块合在一起写。

  先建立脚本文件,作为LQR离线查表的数据,代码如下:

lqr_offline.m

cf=-110000;%侧偏刚度根据曲线估算
cr=cf;%严格来说 CR 和 CF 应该不一样,但差别不大,所以近似认为一样
m=1412;%根据Carsim把整车参数输进去
Iz=1536.7;
a=1.015;
b=2.910-1.015;
k=zeros(5000,4);%申请矩阵k,用来存放LQR的k
for i=1:5000
    vx=0.01*i;%只有vx是变量,所以每隔0.01米每秒,算一下对应的k。
    %这样的话就可涵盖v从 0.01 米每秒到 v 等于 50 米每秒这么大范围内所有的 k 
    A=[0,1,0,0;
        0,(cf+cr)/(m*vx),-(cf+cr)/m,(a*cf-b*cr)/(m*vx);
        0,0,0,1;
        0,(a*cf-b*cr)/(Iz*vx),-(a*cf-b*cr)/Iz,(a*a*cf+b*b*cr)/(Iz*vx)];
    B=[0;
        -cf/m;
        0;
        -a*cf/Iz];
    Q=1*eye(4);
    R=10;
    k(i,:)=lqr(A,B,Q,R);
end
k1=k(:,1)';%建立四个变量,K1、K2、K3、K4,把表记录下来
k2=k(:,2)';%因为Simlink不能传矩阵,只能传行向量,需要做个转置
k3=k(:,3)';
k4=k(:,4)';

  有一种情况是要考虑到,车从零开始加速,启动开始跑,必然有一段速度在 0.01 0.01 0.01 秒之外,从 0 0 0 开始到 0.01 0.01 0.01,再从 0.01 0.01 0.01 到想要的速度,但如果 v x = 0 v_x=0 vx=0,代到 A A A 矩阵里, A A A 就变成奇异阵了,里面有无穷大。所以在 v < 0.01 v<0.01 v<0.01 情况下的 k k k,要进行另外处理,不能用LQR算,后面再讲。

  算出来的 k k k 是由 5000 5000 5000 个行向量组成的矩阵:

在这里插入图片描述

如何查表找和 v x v_x vx 的对应关系呢?

  因为 v x v_x vx 很有规律,从 0.01 、 0.02 、 0.03 0.01、0.02、 0.03 0.010.020.03 一直增大,所以第一行 k k k 必然对应 0.01 m / s 0.01m/s 0.01m/s , 第 2 2 2 行对应 0.02 m / s 0.02m/s 0.02m/s ,以此类推,对应关系非常有规律。

  比如假设速度为 20 m / s 20m/s 20m/s,只要让 20 20 20 除以 0.01 0.01 0.01 即可。

  如果不在里面,比如速度是 20.12345678 20.12345678 20.12345678 这样的小数,对应哪个 k k k,用速度除 0.01 0.01 0.01,除出来是小数而不是整数,那怎么办呢?

  解决方法就是四舍五入,把小数四舍五入变成整数。这样肯定有误差,但误差可忽略不计,因为 0.01 0.01 0.01 秒是非常小的间隔,在两个之间的 k k k 差别不是特别大,所以四舍五入即可。

  LQR 算好之后,就可以写 LQR 模块。LQR 模块由 K 1 、 K 2 、 K 3 、 K 4 K_1、K_2、K_3、 K_4 K1K2K3K4 以及 v x v_x vx 五个变量决定,把 v x v_x vx 输入进去,除以 0.01 0.01 0.01,做四舍五入,看和表的哪组 K 1 、 K 2 、 K 3 、 K 4 K_1、K_2、K_3、 K_4 K1K2K3K4 对应,就把对应的 K 1 、 K 2 、 K 3 、 K 4 K_1、K_2、K_3、 K_4 K1K2K3K4 输出出来,这就是离线 LQR 算法的基本原理。

  离线 LQR 算法程序代码如下:

function k  = fcn(k1,k2,k3,k4,vx)
    if abs(vx)<0.01%特殊处理如果 v 的绝对值小于0.01,
        k=[0,0,0,0];%直接令 k 就等于0,它是航向量
    else
        index=round(vx/0.01);%否则的话就用 v x 除001,再用round四舍五入,值就是所要的匹配的序号
        k=[k1(index),k2(index),k3(index),k4(index)];
    end
end

  把 Q Q Q 设置成了单位矩阵, R R R 100 100 100,可以设置不同的 Q Q Q R R R,算出 k k k 来,再把 k k k 导进去。所以 LQR 和规划算法一样,得先运行跑一遍,让工作区 Workspace 里有 K 1 、 K 2 、 K 3 、 K 4 K_1、K_2、K_3、 K_4 K1K2K3K4,才能进行下一步的 Simulink 仿真。

LQR中的权重矩阵 Q Q Q R R R 该怎么设置呢?

只要记住两句话即可:

  • Q Q Q 越大,算法性能越好,但牺牲了稳定性
    控制量 u u u 可能会特别大,比如可能算出来 u u u 要达到 720 720 720 度这样非常夸张的数值。
  • R R R 越大,控制量越小,过程越平缓
    过程比较平缓比较重要,因为要保证平缓、舒适性、安全性。

  如果无人驾驶车的算法在疯狂打方向盘,看到也害怕,但是代价就是可能跟踪效果不是特别好,所以 Q Q Q R R R 权重的权衡互相矛盾,要具体调合适的 Q Q Q R R R ,期望达到的效果是能很快跟踪上路径,并且控制量 u u u 也不能变化过于剧烈。

  模块写好之后,用 Simlink 拼起来即可。

5、前馈模块实现

  前馈模块需要整车参数、 v x v_x vx 以及投影点的曲率 κ r \kappa_r κr

forward_control

function forword_angle = fcn(vx,a,b,m,cf,cr,k,kr)
    forword_angle=kr*(a+b-b*k(3)-(m*vx*vx/(a+b))*((b/cf)+(a/cr)*k(3)-(a/cr)));
end

  最后把所有结果整合起来即可。

last_angle

function angle = fcn(k,err,forword_angle)
    angle=-k*err+forword_angle;
end

  这样所有代码就都写完了,把它打个包封装成子系统,看起来整洁一点。
在这里插入图片描述

  最后算出来前轮转角,再反馈给 Carsim 的前轮转角即可。

在这里插入图片描述

  注意:单位换算弧度,要化成角度才能输入到 Carsim, Carsim 根据角度控制,不是根据弧度控制的。

6、避免代数环的延迟模块

  在算出角度的线上再加延迟模块,作用是如果输入是这一时刻的值,输出是上时刻的值,它就等于延迟单元。这是为了避免代数环,即输入直接决定输出,输出又直接决定输入,会导致套娃式循环。所以为了打破循环就要加延迟模块来避免代数环。

  需要整车状态 ( x , y , φ , v x , v y , φ ˙ ) (x,y,\varphi ,v_x,v_y,\dot{\varphi}) (x,y,φ,vx,vy,φ˙) 作为输入才能算出角度, x , y , φ x,y,\varphi x,y,φ 又需要角度作为输入打方向盘, x , y , φ x,y,\varphi x,y,φ 肯定也都变了,也就是说车辆状态决定角度,又决定车辆状态,他们之间就会变成套娃循环。所以说要加延迟模块,避免代数环现象。

7、仿真测试

  后轮转角给 0 0 0,节气门开度给 0.15 0.15 0.15,也就是给一点点油门,模型只做了横向控制,控制方向盘转角,并没有做纵向控制。纵向控制就是控制油门的节气门开度,后续再讲,所以说模型只搭了一半,在这里并不控制纵向速度,就给它恒定油门 0.15 0.15 0.15 ,运行程序看看效果,用 X Y XY XY 观测器,观察算法结果:

在这里插入图片描述
  可能觉得是个椭圆而不是圆,这是因为横坐标和纵坐标间隔不完全一样,横坐标要宽一点,纵坐标窄一点,所以看起来不太像是圆,而像是椭圆,实际上应该是个圆。

  其实很明显看到车转弯半径肯定超过 5 5 5 米,因为道路确实比较严苛。再跑一遍,看算出来方向盘转角是什么情况。

在这里插入图片描述

  看一下算出来的前端转角,其实是不太好的,有的地方在不停鬼畜,不停地抖来抖去,其实不是一件好事情。

  在有的地方突变太厉害,突然猛的打一下方向盘,在转弯时打得非常快。算法写出来有利有弊,方向盘打的太快了,而且方向盘在转弯时,不停的鬼畜抖来抖去,这是无法接受的。

  但是也有好的地方,至少控制的跟踪效果做出来了,确实是按照所规划的轨迹在跑,而且规划的轨迹是比较严苛的。

  转弯半径很小,而且有很多很急的弯,结果也算勉强能跟上,在没有做纵向控制且在不停加速的情况下,仍然勉强能跟上规划的轨迹,所以有一定的控制效果,但舒适性并不是特别好,方向盘在不停抖来抖去。

为什么会导致方向盘在鬼畜?而且在转弯时,方向盘会猛的突变?

  这是有原因的,在下一节会讲算法性能不好的原因以及怎么改进。


五、总结

  本节博客详细介绍了自动驾驶控制算法的建模过程,涵盖了从Carsim设置到模型搭建的各个方面。通过配置Carsim软件,获取整车参数并计算侧偏刚度,为后续的控制算法提供基础数据。

  通过编写路径规划代码,生成所需的轨迹信息。在Matlab中,搭建了控制模型的各个模块,包括预测模块、误差和曲率计算模块、AB模块和LQR模块、前馈控制模块等。

  最后,将模型整合并进行了仿真测试,展示了控制算法对规划轨迹的跟踪效果。

  大家不妨在这一节照着思路把模型搭好,在下一节就会告诉各位怎么调模型,使其更平顺。欢迎关注后续内容!


参考资料

  【基础】自动驾驶控制算法第八讲(三) 代码与模型


后记:

🌟 感谢您耐心阅读这篇关于 自动驾驶横向控制算法代码与模型 的技术博客。 📚

🎯 如果您觉得这篇博客对您有所帮助,请不要吝啬您的点赞和评论 📢

🌟您的支持是我继续创作的动力。同时,别忘了收藏本篇博客,以便日后随时查阅。🚀

🚗 让我们一起期待更多的技术分享,共同探索移动机器人的无限可能!💡

🎭感谢您的支持与关注,让我们一起在知识的海洋中砥砺前行 🚀

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

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

相关文章

以太网--TCP/IP协议(一)

概述 以太网是局域网的一种&#xff0c;其他的比如还有令牌环、FDDI。和局域网对应的就是广域网&#xff0c;如Internet&#xff0c;城域网等。 从网络层次看&#xff0c;局域网协议主要偏重于低层&#xff08;业内一般把物理层、数据链路层归为低层&#xff09;。以太网协议…

单片机毕业设计基于单片机的智能门禁系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍程序代码部分参考 设计清单具体实现截图参考文献设计获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师&#xff0c;一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP…

vue动态统计图的绘画

效果图&#xff1a; 实现&#xff1a; 一、导入依赖 import echarts from echarts 二、vue的代码实现 1.在main.js导入文件 // 引入 echarts 插件 import echarts from echarts // 配置成全局组件 Vue.prototype.$echarts echarts2.代码实现 <template><!--为echa…

韩国火烧车影响出现,浙江出现限制电车进入地下车库,车主难受了

韩国电动汽车起火&#xff0c;烧毁140辆汽车&#xff0c;还导致大楼损坏以及居民受伤的后果&#xff0c;如今在中国市场也产生了影响&#xff0c;《华商报》旗下的《大风新闻》报道指&#xff0c;浙江多地的饭店、大厦禁止电动汽车进入地下车库&#xff0c;这下子电动汽车车主又…

滑动窗口在算法中的应用

滑动窗口是一种经典的算法技巧&#xff0c;就像在处理一系列动态数据时&#xff0c;用一扇可以滑动的“窗口”来捕捉一段连续的子数组或子字符串。通过不断地移动窗口的起点或终点&#xff0c;我们能够以较低的时间复杂度来解决一系列问题。在这篇文章中&#xff0c;我们将通过…

图形视频处理软件Adobe After Effects(AE)2024WIN/MAC下载及系统要求

目录 一、Adobe AE软件简介 1.1 什么是Adobe AE软件 1.2 AE软件的发展历程 1.3 AE软件的应用领域 二、Adobe AE软件下载 2.1 下载 2.2 下载注意事项 三、Adobe AE软件系统要求 3.1 最低配置要求 3.2 推荐配置要求 3.3 显示器和分辨率 四、Adobe AE软件安装与使用 …

【MacOS】mac定位服务中删除已经卸载的软件

mac定位服务中删除已经卸载的软件 网上的帖子真不靠谱 直接右键 WeTypeSettings &#xff0c;查找位置&#xff0c;丢废纸篓即可&#xff01;会提示你卸载的&#xff01;

Pyramid: Real-Time LoRa Collision Decoding with Peak Tracking技术思考与解读

一点点个人的论文解读、技术理解&#xff0c;难免会有错误&#xff0c;欢迎大家一起交流和学习~~ &#x1f600;作者关于lora的系列文章从问题陈述到方法论的提出&#xff0c;再到实验评估&#xff0c;文章结构条理清晰&#xff0c;逻辑性强&#xff0c;并深入分析了LoRa信号处…

力扣刷题(5)

整数转罗马数字 整数转罗马数字-力扣 思路&#xff1a; 把各十百千位可能出现的情况都列出来&#xff0c;写成一个二维数组找出该数的各十百千位&#xff0c;与数组中的罗马元素对应 const char* ch[4][10]{{"", "I", "II", "III"…

webpack - 五大核心概念和基本配置(打包一个简单HTML页面)

// 五大核心概念 1. entry&#xff08;入口&#xff09; 指示Webpack从哪个文件开始打包2. output&#xff08;输出&#xff09; 指示Webpack打包完的文件输出到哪里去&#xff0c;如何命名等3. loader&#xff08;加载器&#xff09; webpack本身只能处理js&#xff0c;json等…

Bev pool 加速(2):自定义c++扩展

文章目录 1. c++扩展2. 案例2.1 案例12. 1.1 代码实现(1) c++ 文件(2) setup.py编写(3) python 代码编写2.1 案例1在bevfusion论文中,将bev_pooling定义为view transform中的效率瓶颈,bevfusion 主要就是对bev_pooling进行了加速,使得视图转换的速度提高了40倍,延迟从500ms…

charles配置安卓抓包(避坑版)

下载Charleshttps://www.charlesproxy.com/安装&#xff0c;疯狂点击下一步即可注册&#xff1a;打开Charles&#xff0c;选择“Help”菜单中的“Register Charles”&#xff0c;进网站生成密钥&#xff1a;https://www.zzzmode.com/mytools/charles/,将生成的密钥填入注册重启…

JavaScript练手小技巧:利用鼠标滚轮控制图片轮播

近日&#xff0c;在浏览网站的时候&#xff0c;发现了一个有意思的效果&#xff1a;一个图片轮播&#xff0c;通过上下滚动鼠标滚轮控制图片的上下切换。 于是就有了自己做一个的想法&#xff0c;顺带复习下鼠标滚轮事件。 鼠标滚轮事件&#xff0c;参考这篇文章&#xff1a;…

Vue 3 + Element Plus 封装单列控制编辑的可编辑表格组件

在Web应用开发中&#xff0c;经常需要提供表格数据的编辑功能。本文将介绍如何使用Vue 3结合Element Plus库来实现一个支持单列控制编辑功能的表格&#xff0c;并通过封装组件的形式提高代码的复用性。通过本教程&#xff0c;你将学会如何构建一个具备单列控制编辑功能的表格组…

Cloudways搭建WordPress外贸独立站完整教程(1)

验证邮件发送完成后&#xff0c;就等待Cloudways的回复邮件&#xff0c;一般24小时之内就会收到激活的邮件。 Cloudways账号升级 激活成功后还需要账户升级&#xff0c;Cloudways提供了为期3天的免费试用体验。如果在试用期结束之前未绑定信用卡以升级账户&#xff0c;试用期…

UE5学习笔记21-武器的射击功能

一、创建C类 创建武器子弹的类&#xff0c;创建生产武器子弹的类&#xff0c;创建弹壳的类&#xff0c;生产武器子弹的类的父类是武器的类 创建后如图&#xff0c;ProjectileMyWeapon类(产生子弹的类)继承自weapon类&#xff0c;Projectile(子弹的类)&#xff0c;Casing(弹壳声…

Claude 3.5:如何高效辅助编程——全面入门指南

在现代编程世界中&#xff0c;AI的角色越来越重要&#xff0c;尤其是在代码生成、调试、文档生成等领域中&#xff0c;AI工具的运用让开发者可以更高效地完成任务。Claude 3.5是一个这样的AI助手&#xff0c;凭借其强大的自然语言处理能力&#xff0c;在编程中提供了大量的支持…

Sui Narwhal and Tusk 共识协议笔记

一、Overwiew [ 整体流程: Client提交transaction到Narwhal Mempool。(Narwhal Mempool由一组worker和一个primary组成) Mempool接收到的Transaction->以Certificate的形式进行广播 由worker将交易打包为Batch,worker将Batch的hash发送给primary primary上运行了mempo…

mysql笔记4(数据类型)

数据库的数据类型应该是数据库架构师(DBA)和产品经理沟通后依据公司的项目、业务而定的&#xff0c;而且会不停地变化。数据类型的选择方面没有一个统一的标准&#xff0c;但是应该符合业务、项目的逻辑标准。 菜鸟教程 Mysql 数据类型 文章目录 1. int类型2. 浮点数3. 定点数4…

C# Dotfuscator加密dll设置流程

按照以下步骤处理后&#xff0c;反编译基本只能看到函数名&#xff0c;看不到源代码 1.Input 2.Setting 3.Rename 4.Rename 5.Control Flow 6.String Encryption 7.Output