(4)【轨迹优化篇】方法一:基于Frenet车道线坐标系,采用解耦采样五次多项式拟合进行局部规划

news2025/1/9 14:24:59

系列文章目录

提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加
TODO:写完再整理

文章目录

  • 系列文章目录
  • 前言
  • A、第一章:Frenet车道线坐标系介绍及坐标转换
    • 一、frenet坐标系介绍
      • 1.Frenet坐标的定义
      • 2.ST图(纵向速度规划)
      • 3.SL图(横向位移规划)
      • 4.frenet坐标系的横纵向误差定义
    • 二、坐标系相互转换的方法
      • 1.笛卡尔坐标转换成frenet坐标系
        • (1)把轨迹投影到frenet坐标系的st图
        • (2)把障碍物投影到frenet坐标系的st图
        • (3)把世界坐标系下的xy坐标转换成车坐标系的sl坐标系
      • 2.Frenet坐标系转笛卡尔坐标系
      • 3.坐标转换的程序实现
    • 总结
  • B、第二章:横纵向解耦采样生成路径+Jerk轨迹优化算法in Frenet Frame
    • 【核心思想】基于Frenet坐标系的无人车轨迹生成并优化的功能
    • 基于Frenet坐标系方法生成的路径和直接用三五次样条(三五次多项式)生成的路径的不同之处
    • 一、无人驾驶最优轨迹序列的定义
      • 1.横向控制最优轨迹序列
      • 2.纵向控制最优轨迹序列
      • 3.Frenet坐标系定义
      • 4.为什么要用Frenet坐标系做轨迹优化
    • 二、基于Frenet坐标系轨迹优化的过程理论
      • 1.步骤一:设计Jerk最优化问题
      • 2.步骤二:求解Jerk最优化问题(即求解一个5次多项式的5各系数)
      • 3.步骤三:确定最优的轨迹
        • 第一步:通过一组目标配置来求得轨迹的备选集合
          • 横向轨迹的备选集合
          • 纵向轨迹的备选集合
        • 第二步:备选集合中基于Jerk最小化的原则,设计代价损失函数来选择最优轨迹
          • 横向轨迹的损失函数设计
          • 横向轨迹 + 纵向轨迹的损失函数设计
        • 第三步:通过限幅剔除轨迹(事故避免)
    • 三、基于Frenet坐标系轨迹优化的过程代码实现
      • 1.首先,我们生成要追踪的参考线以及静态障碍物
      • 2.定义一些参数
      • 3.使用基于Frenet的优化轨迹方法生成一系列横向和纵向的轨迹,并且计算每条轨迹对应的损失
      • 4.五次多项式(五次样条曲线求解)类
    • 四、障碍物规避实现
    • 总结
    • 参考资料


前言

认知有限,望大家多多包涵,有什么问题也希望能够与大家多交流,共同成长!

本文先对基于Frenet坐标系轨迹优化,车道线解耦采样五次多项式拟合做个简单的介绍,具体内容后续再更,其他模块可以参考去我其他文章


提示:以下是本篇文章正文内容

A、第一章:Frenet车道线坐标系介绍及坐标转换

一、frenet坐标系介绍

1.Frenet坐标的定义

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
Frenet坐标系描述了汽车相对于道路的位置,S代表沿道路的距离,也称为纵坐标,L表示与纵向线的位移,也被称为横坐标,纵坐标表示在道路中行驶距离,横坐标表示汽车偏离中心线的距离

通过Frenet坐标可将障碍物与轨迹投射到S纵轴和L横轴上,目标是将三维轨迹(纵向维度、横向维度和时间维度)分解成两个单独的二维问题,即通过分离轨迹的纵向和横向分量来解决,一个二维轨迹是指具有时间戳的纵向轨迹(ST轨迹),另一个二维轨迹相对于纵向轨迹的横向偏移(SL轨迹)。

frenet坐标系是比较针对车道(参考线)线场景的【s轴与参考线平行,L轴与参考线垂直】,在车道线场景时才专门用到
.

2.ST图(纵向速度规划)

ST图可以帮助我们设计和选择速度曲线,“s”表示车辆的纵向位移,“t”表示时间,ST图上的曲线是对车辆运动的描述,因为它说明了车辆在不同时间的位置,由于速度是位置变化的速率,因此可以通过曲线的斜率从ST图上推断速度,斜坡越陡说明在短时间内有更大的移动对应更快的速度
在这里插入图片描述

在ST图内可以将障碍物绘制为在特定时间段内阻挡道路部分的矩形,速度曲线不得与矩形相交

ST轨迹的终止状态:根据情况可以将状态分为三组巡航、跟随与停止。

.

3.SL图(横向位移规划)

SL图表示了车辆在不同的纵坐标上的横向位移,反映在不同纵向
在这里插入图片描述
若在高速的场景中
在这里插入图片描述

.

4.frenet坐标系的横纵向误差定义

在这里插入图片描述

.

二、坐标系相互转换的方法

对于规划,需要从笛卡尔坐标系转换到frenet坐标系
对于控制、地图、定位等模块,需要从frenet坐标系转换到笛卡尔坐标系

1.笛卡尔坐标转换成frenet坐标系

(1)把轨迹投影到frenet坐标系的st图

在这里插入图片描述
在这里插入图片描述

(2)把障碍物投影到frenet坐标系的st图

在这里插入图片描述
在这里插入图片描述

(3)把世界坐标系下的xy坐标转换成车坐标系的sl坐标系

在这里插入图片描述
sl坐标系转xy坐标系是唯一的,xy坐标系转sl坐标系不是唯一的,但是有约束的时候可以是唯一的
xy坐标系转sl坐标系的转换过程相对复杂,推导不要去推,记住转换结果会用就好

.
.

2.Frenet坐标系转笛卡尔坐标系

一旦同时拥有了ST和SL轨迹,就需要将它们重新转化为笛卡尔坐标,然后将它们相结合,构建由二维路径点和一维时间戳组成的三维轨迹,ST轨迹是随时间变化的纵向位移,SL纵向轨迹上每个点的横向偏移,两条轨迹都有纵坐标S,可以通过将S值进行匹配来合并轨迹
在这里插入图片描述

3.坐标转换的程序实现

参考百度的demo

总结

em planner和lattice planner 都有用到frenet坐标系,所以有必要了解一下


B、第二章:横纵向解耦采样生成路径+Jerk轨迹优化算法in Frenet Frame

【核心思想】基于Frenet坐标系的无人车轨迹生成并优化的功能

功能
给定地图上的起点和目标点,通过mini_jerk或者mini_snap的插值优化问题,求解生成出两点间的轨迹,并保证了两点间的轨迹最优的
.

基于Frenet坐标系方法生成的路径和直接用三五次样条(三五次多项式)生成的路径的不同之处

直接用三次五次样条(三五次多项式)生成的路径不考虑实际场景,没有代价损失函数,因此没有避障等功能
三次样条插值生成轨迹方法,参考我另外一篇博客
https://blog.csdn.net/qq_35635374/article/details/121772237

.
基于Frenet坐标系方法生成的路径是继承三五次样条的方法生成路径方法的,通过尝试给定sd横纵两个方向的目标点,使用三五次样条的方法生成路径一堆候选路径,再通过求解mini_jerk的优化问题和设计不同代价损失函数项,限幅等选择出最优轨迹
在这里插入图片描述

.
.

一、无人驾驶最优轨迹序列的定义

Frenet坐标系分为横向和纵向两个坐标轴,因为讨论最优轨迹序列应该分成横向控制最优轨迹序列、纵向控制最优轨迹序列

1.横向控制最优轨迹序列

对于横向控制而言,假定由于车辆因为之前躲避障碍物或者变道或者其他制动原因而偏离了期望的车道线,那么此时最优的动作序列(或者说轨迹)是在车辆制动能力的限制下,相对最安全,舒适,简单和高效的轨迹。
.
.

2.纵向控制最优轨迹序列

纵向的最优轨迹也可以这么定义:如果车辆此时过快,或者太接近前方车辆,那么就必须做减速,那么具体什么是“舒适而又简单的”减速呢?我们可以使用 Jerk 这个物理量来描述,Jerk即加速度的变化率,也即加加速度,通常来说,过高的加加速度会会引起乘坐者的不适,所以,从乘坐舒适性而言,应当优化Jerk这个量,同时,引入轨迹的制动周期 T , 即一个制动的操作时间:
在这里插入图片描述
.
.

3.Frenet坐标系定义

在Frenet坐标系中,我们使用道路的中心线作为参考线,使用参考线的切线向量 tt 和法线向量 nn 建立一个坐标系,如下图的右图所示,这个坐标系即为Frenet坐标系,它以车辆自身为原点,坐标轴相互垂直,分为 ss 方向(即沿着参考线的方向,通常被称为纵向,Longitudinal)和 dd 方向(即参考线当前的法向,被称为横向,Lateral),相比于笛卡尔坐标系(下图的作图),Frenet坐标系明显地简化了问题,因为在公路行驶中,我们总是能够简单的找到道路的参考线(即道路的中心线),那么基于参考线的位置的表示就可以简单的使用纵向距离(即沿着道路方向的距离)和横向距离(即偏离参考线的距离)来描述,同样的,两个方向的速度(s˙s˙ 和 d˙d˙)的计算也相对简单。
在这里插入图片描述
那么现在我们的动作规划问题中的配置空间就一共有三个维度:(s,d,t) , t 是我们规划出来的每一个动作的时间点,轨迹和路径的本质区别就是轨迹考虑了时间这一维度。

.
.

4.为什么要用Frenet坐标系做轨迹优化

答案:将动作规划这一高维度的优化问题分割成横向和纵向两个方向上的彼此独立的优化问题
在这里插入图片描述
假若规划层中的任务层下达变道的任务,要求当前车辆在 t8t8 越过虚线完成一次变道,即车辆在横向上需要完成一个 ΔdΔd 以及纵向上完成一个 ΔsΔs的移动,则可以将 s和 d 分别表示为关于 t 的函数:s(t)和 d(t)(上图右图),那么 d,sd,s 关于时间 tt 的最优轨迹应该选择哪一条呢?通过这种转换原来的动作规划问题被分割成了两个独立的优化问题,对于横向和纵向的轨迹优化,我们选取损失函数 C ,将使得 C 最小的轨迹作为最终规划的动作序列。

简单来说,就是优化一个具有二维信息的轨迹(x,y),转换成优化两条独立的一维轨迹s(t)和 d(t)
.
.

二、基于Frenet坐标系轨迹优化的过程理论

Werling方法中损失函数的定义使用了加加速度 Jerk 函数。Jerk即加速度的变化率,通常来说,绝对值过高的加加速度会会引起乘坐者的不适,所以,从乘坐舒适性而言,应当优化Jerk这个量

1.步骤一:设计Jerk最优化问题

由于我们将轨迹优化问题分割成了 s 和 d 两个方向,所以Jerk最小化可以分别从横向和纵向进行,令 p 为我们考量的配置(即 s 或 d),加加速度Jt关于配置 p 在时间段 t1−t0 内累计的Jerk的表达式为:

在这里插入图片描述
现在我们的任务是找出能够使得 Jt(p(t)) 最小的 p(t)
怎么理解上面这个优化问题呢?
上述最优问题表达式是在一段短时间段内对加速度的积分,若加速度曲线震荡,加速度的积分在较短的时间段每的积分至就大,我们的目的是通过优化位置p(t)使得加加速度不震荡
.
.

2.步骤二:求解Jerk最优化问题(即求解一个5次多项式的5各系数)

Takahashi的文章——Local path planning and motion control for AGV in positioning中已经证明,任何Jerk最优化问题中的解都可以使用一个5次多项式来表示:
在这里插入图片描述
要解这个方程组需要一些初始配置和目标配置,以横向路径规划为例,初始配置为 D0=[d0,d0˙,d0¨],即 t0t0 时刻车辆的横向偏移,横向速度和横向加速度为 d0,d0˙,d0¨,即可得方程组:

在这里插入图片描述
同理,根据横向的目标配置 D1=[d1,d1˙,d1¨] 可得方程组:
在这里插入图片描述
我们通过令 t0=0来简化这个六元方程组的求解,可直接求得 αd0 ,αd1 和 αd2 为:
在这里插入图片描述
令 T=t1−t0 ,剩余的三个系数 αd3,αd4,αd5 ,可通过解如下矩阵方程得到:
在这里插入图片描述
该方程的解可以通过Python的Numpy中的 np.linalg.solve 简单求得。

至此,我们在给定任意的初始配置 D0=[d0,d0˙,d0¨],目标配置 D1=[d1,d1˙,d1¨]以及制动时间 T 的情况下,可以求的对应的 dd 方向关于时间 t 的五次多项式的系数

为了区分横向和纵向,我们使用 αdi 和 αsi 来分别表示d和s方向的多项式系数,同理,可以使用相同的方法来求解纵向(即 s 方向)的五次多项式系数。

.
.

3.步骤三:确定最优的轨迹

第一步:通过一组目标配置来求得轨迹的备选集合

横向轨迹的备选集合

我们仍然以 d 方向的优化轨迹为例讲解:
我们可以取如下目标配置集合来计算出一组备选的多项式集合
在这里插入图片描述
对于优化问题而言,我们实际上希望车辆最终沿着参考线(道路中心线)平行的方向行驶,所以我们令 di˙=di¨=0di˙=di¨=0 ,那么目标配置只涉及 didi 和 TjTj 两个变量的组合,而这两个变量在无人驾驶的应用场景中实际上是受限的,我们可以通过定义 (dmin,dmax)(dmin,dmax) 和 (Tmin,Tmax)(Tmin,Tmax) 来约束目标配置的取值范围,通过 ΔdΔd 和 ΔTΔT 来限制采样密度,从而在每一个制动周期获得一个有限的备选轨迹集合
在这里插入图片描述
.

纵向轨迹的备选集合

在不同的场景下纵向轨迹的优化的损失函数也各不相同,Werling方法中将纵向轨迹的优化场景大致分成如下三类:

  • 跟车
  • 汇流和停车
  • 车速保持

在本文中我们详细了解车速保持场景下的纵向轨迹优化,在高速公路等应用场景中,目标配置中并不需要考虑目标位置(即 s1s1 ),所以在该场景下,目标配置仍然是 (s0,s0˙,s0¨),目标配置变成了 (s1˙,s1¨),损失函数为:
在这里插入图片描述
其中 sc˙ 是我们想要保持的纵向速度,第三个惩罚项的引入实际上是为了让目标配置中的纵向速度尽可能接近设定速度,该情景下的目标配置集为:
在这里插入图片描述
即优化过程中的可变参数为 Δsi˙ 和 Tj ,同样,也可以通过设置 ΔT 和 ΔΔsi˙ 来设置轨迹采样的密度,从而获得一个有限的纵向轨迹集合:
在这里插入图片描述
.
.

第二步:备选集合中基于Jerk最小化的原则,设计代价损失函数来选择最优轨迹

要在备选集合中选择最优轨迹(即上图中的绿色轨迹),我们需要设计损失函数,对于不同的场景,损失函数也不相同

横向轨迹的损失函数设计

在较高速度的情况下,横向轨迹的损失函数为:
在这里插入图片描述
该损失函数包含三个惩罚项:

  • kjJt(d(t)):惩罚Jerk大的备选轨迹;
  • ktT :制动应当迅速,时间短;
  • kdd :目标状态不应偏离道路中心线太远

其中 kj,kt和 kd 是这三个惩罚项的系数,它们的比值大小决定了我们的损失函数更加注重哪一个方面的优化,由此我们可以算出所有备选轨迹的损失,取损失最小的备选轨迹作为我们最终的横向轨迹。

值得注意的是,以上的损失函数仅适用于相对高速度的场景,在极端低速的情况下,车辆的制动能力是不完整的,我们不再将d表示为关于时间t的五次多项式,损失函数也会略有不同,但是这种基于有限采样轨迹,通过优化损失函数搜索最优轨迹的方法仍然是一样的,在此不再赘述。

横向轨迹 + 纵向轨迹的损失函数设计

。以上我们分别讨论了横向和纵向的最优轨迹搜索方法,在应用中,我们将两个方向的损失函数合并为一个,即:
在这里插入图片描述
这样,我们就可以通过最小化 Ctotal 得到优化轨迹集合(我们不能得到“最优”的轨迹多项式参数,还可以得到“次优”,“次次优”轨迹等等)。
.
.

第三步:通过限幅剔除轨迹(事故避免)

们上面的轨迹优化损失函数中并没有包含关于障碍物躲避的相关惩罚,并且我们的损失函数中也没有包含最大速度,最大加速度和最大曲率等制动限制,也就是说我们的优化轨迹集合并没有考虑障碍物规避和制动限制因素,不将障碍物避免加入到损失函数中的一个重要的原因在于碰撞惩罚项的引入将代入大量需要人工调整的参数(即权重),是的损失函数的设计变得复杂 ,Werling方法将这些因素的考量独立出来,在完成优化轨迹以后进行。具体来说,我们会在完成所有备选轨迹的损失计算以后进行一次轨迹检查,过滤掉不符合制动限制的,可能碰撞障碍物的轨迹,检查内容包括:

s方向上的速度是否超过设定的最大限速
s方向的加速度是否超过设定的最大加速度
轨迹的曲率是否超过最大曲率
轨迹是否会引起碰撞(事故)

通常来说,障碍物规避又和目标行为预测等有关联,本身即使一个复杂的课题,高级自动驾驶系统通常具备对目标行为的预测能力,从而确定轨迹是否会发生事故。在本节中,我们关注的重点是无人车的动作规划,故后面的实例仅涉及静态障碍物的规避和动作规划。
.
.

三、基于Frenet坐标系轨迹优化的过程代码实现

完整代码链接:
https://download.csdn.net/download/adamshan/10494062

1.首先,我们生成要追踪的参考线以及静态障碍物

# 路线
wx = [0.0, 10.0, 20.5, 30.0, 40.5, 50.0, 60.0]
wy = [0.0, -4.0, 1.0, 6.5, 8.0, 10.0, 6.0]
# 障碍物列表
ob = np.array([[20.0, 10.0],
               [30.0, 6.0],
               [30.0, 5.0],
               [35.0, 7.0],
               [50.0, 12.0]
               ])

tx, ty, tyaw, tc, csp = generate_target_course(wx, wy)

生成如下参考路径以及障碍物,其中红线就是我们的全局路径,蓝点为障碍物
在这里插入图片描述

2.定义一些参数

# 参数
MAX_SPEED = 50.0 / 3.6  # 最大速度 [m/s]
MAX_ACCEL = 2.0  # 最大加速度[m/ss]
MAX_CURVATURE = 1.0  # 最大曲率 [1/m]
MAX_ROAD_WIDTH = 7.0  # 最大道路宽度 [m]
D_ROAD_W = 1.0  # 道路宽度采样间隔 [m]
DT = 0.2  # Delta T [s]
MAXT = 5.0  # 最大预测时间 [s]
MINT = 4.0  # 最小预测时间 [s]
TARGET_SPEED = 30.0 / 3.6  # 目标速度(即纵向的速度保持) [m/s]
D_T_S = 5.0 / 3.6  # 目标速度采样间隔 [m/s]
N_S_SAMPLE = 1  # 目标速度的采样数量
ROBOT_RADIUS = 2.0  # robot radius [m]

# 损失函数权重
KJ = 0.1
KT = 0.1
KD = 1.0
KLAT = 1.0
KLON = 1.0

3.使用基于Frenet的优化轨迹方法生成一系列横向和纵向的轨迹,并且计算每条轨迹对应的损失

def calc_frenet_paths(c_speed, c_d, c_d_d, c_d_dd, s0):
    frenet_paths = []

    # 采样,并对每一个目标配置生成轨迹
    for di in np.arange(-MAX_ROAD_WIDTH, MAX_ROAD_WIDTH, D_ROAD_W):

        # 横向动作规划
        for Ti in np.arange(MINT, MAXT, DT):
            fp = Frenet_path()
            # 计算出关于目标配置di,Ti的横向多项式
            lat_qp = quintic_polynomial(c_d, c_d_d, c_d_dd, di, 0.0, 0.0, Ti)

            fp.t = [t for t in np.arange(0.0, Ti, DT)]
            fp.d = [lat_qp.calc_point(t) for t in fp.t]
            fp.d_d = [lat_qp.calc_first_derivative(t) for t in fp.t]
            fp.d_dd = [lat_qp.calc_second_derivative(t) for t in fp.t]
            fp.d_ddd = [lat_qp.calc_third_derivative(t) for t in fp.t]

            # 纵向速度规划 (速度保持)
            for tv in np.arange(TARGET_SPEED - D_T_S * N_S_SAMPLE, TARGET_SPEED + D_T_S * N_S_SAMPLE, D_T_S):
                tfp = copy.deepcopy(fp)
                lon_qp = quartic_polynomial(s0, c_speed, 0.0, tv, 0.0, Ti)

                tfp.s = [lon_qp.calc_point(t) for t in fp.t]
                tfp.s_d = [lon_qp.calc_first_derivative(t) for t in fp.t]
                tfp.s_dd = [lon_qp.calc_second_derivative(t) for t in fp.t]
                tfp.s_ddd = [lon_qp.calc_third_derivative(t) for t in fp.t]

                Jp = sum(np.power(tfp.d_ddd, 2))  # square of jerk
                Js = sum(np.power(tfp.s_ddd, 2))  # square of jerk

                # square of diff from target speed
                ds = (TARGET_SPEED - tfp.s_d[-1]) ** 2
                # 横向的损失函数
                tfp.cd = KJ * Jp + KT * Ti + KD * tfp.d[-1] ** 2
                # 纵向的损失函数
                tfp.cv = KJ * Js + KT * Ti + KD * ds
                # 总的损失函数为d 和 s方向的损失函数乘对应的系数相加
                tfp.cf = KLAT * tfp.cd + KLON * tfp.cv

                frenet_paths.append(tfp)

    return frenet_paths

.
.

4.五次多项式(五次样条曲线求解)类

这里的五次多项式的系数的求解过程和我们前面的理论讲解是一样的,只不过我们使用Numpy中的 np.linalg.solve(A, b) 方法将矩阵解了出来。

class quintic_polynomial:
    def __init__(self, xs, vxs, axs, xe, vxe, axe, T):
        # 计算五次多项式系数
        self.xs = xs
        self.vxs = vxs
        self.axs = axs
        self.xe = xe
        self.vxe = vxe
        self.axe = axe

        self.a0 = xs
        self.a1 = vxs
        self.a2 = axs / 2.0

        A = np.array([[T ** 3, T ** 4, T ** 5],
                      [3 * T ** 2, 4 * T ** 3, 5 * T ** 4],
                      [6 * T, 12 * T ** 2, 20 * T ** 3]])
        b = np.array([xe - self.a0 - self.a1 * T - self.a2 * T ** 2,
                      vxe - self.a1 - 2 * self.a2 * T,
                      axe - 2 * self.a2])
        x = np.linalg.solve(A, b)

        self.a3 = x[0]
        self.a4 = x[1]
        self.a5 = x[2]

    def calc_point(self, t):
        xt = self.a0 + self.a1 * t + self.a2 * t ** 2 + \
             self.a3 * t ** 3 + self.a4 * t ** 4 + self.a5 * t ** 5

        return xt

    def calc_first_derivative(self, t):
        xt = self.a1 + 2 * self.a2 * t + \
             3 * self.a3 * t ** 2 + 4 * self.a4 * t ** 3 + 5 * self.a5 * t ** 4

        return xt

    def calc_second_derivative(self, t):
        xt = 2 * self.a2 + 6 * self.a3 * t + 12 * self.a4 * t ** 2 + 20 * self.a5 * t ** 3

        return xt

    def calc_third_derivative(self, t):
        xt = 6 * self.a3 + 24 * self.a4 * t + 60 * self.a5 * t ** 2

        return xt

.
.

四、障碍物规避实现

def check_paths(fplist, ob):
    okind = []
    for i in range(len(fplist)):
        if any([v > MAX_SPEED for v in fplist[i].s_d]):  # 最大速度检查
            continue
        elif any([abs(a) > MAX_ACCEL for a in fplist[i].s_dd]):  # 最大加速度检查
            continue
        elif any([abs(c) > MAX_CURVATURE for c in fplist[i].c]):  # 最大曲率检查
            continue
        elif not check_collision(fplist[i], ob):
            continue

        okind.append(i)

    return [fplist[i] for i in okind]

由此可以看出,最终的优化轨迹的选择并不单纯基于最小损失函数,轨迹检查还会过滤掉一些轨迹,所以使用基于Frenet的优化轨迹来做无人车的动作规划,通常能够找到有限集的最优解,当最优解无法通过检查是,自会采用“次优解”甚至更加“次优的”解。


总结

1、基于Frenet坐标系的无人车轨迹生成并优化的功能:
给定地图上的起点和目标点,通过mini_jerk或者mini_snap的插值优化问题,求解出两点间的轨迹,并保证了两点间的轨迹最优的

.
2、基于Frenet坐标系的无人车轨迹生成并优化的使用方法
可以把轨迹从笛卡尔坐标系转换到frenet坐标系,完成基于Frenet坐标系的无人车轨迹优化后,把轨迹再转换到笛卡尔坐标系
.
3、基于Frenet坐标系的无人车轨迹生成并优化是继承五次样条(五次多项式曲线)生成路径的方法的
.
.
4、基于Frenet坐标系计算五次样条的方法和基于笛卡尔坐标系计算五次样条的方法异同
基于笛卡尔坐标系计算五次样条的方法,可以看我这篇博客
https://blog.csdn.net/qq_35635374/article/details/121772237
https://blog.csdn.net/qq_35635374/article/details/120732771

基于笛卡尔坐标系计算五次样条的方法的轨迹是是二维的,即横纵坐标是xy-- y(x)
基于Frenet坐标系计算五次样条的方法的轨迹是把一维(把二维轨迹解耦至两个一维轨迹),即s(t)、d(t)

参考资料

https://blog.csdn.net/AdamShan/article/details/80779615

理论可以参考我下面这篇博客
http://t.csdn.cn/hU2Wl

.

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

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

相关文章

【ARM Coresight 系列文章 3 - ARM Coresight 组件 DAP(Debug Access Port) 介绍】

文章目录 1.1 Debug Access Port1.2 Access Port1.2.1 IDR 寄存器 1.3 Mem-APs 介绍1.3.1 Debug 寄存器访问模型1.3.2 APs 中寄存器的介绍 1.1 Debug Access Port 外部 Debugger(DS-5/Trace32) 会通过JTAG接口或者SWD接口和DAP相连: JITAG 一般是5个pin&#xff1…

JAVA每日一练(1)

【程序1】 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子对数为多少? import java.util.Scanner;/*…

【ElasticSearch】数据聚合语法与Java实现

文章目录 1、聚合的分类2、DSL实现bucket聚合3、DSL实现Metrics 聚合4、RestClient实现聚合5、需求:返回过滤条件的信息6、带过滤条件的聚合 1、聚合的分类 聚合(aggregations)可以实现对文档数据的统计、分析、运算。(类比MySQL…

如何记牢托福口语考试的关键词?

一般情况下,托福独立口语一类问题是自由回答间题(Free-choice Response),如:If you could have any job in the world, what would it be? Use details to support your. response;另一类是选择类问题(Paired-choice Response),如…

BERT论文解读及实现(一)

BERT: Pre-training of Deep Bidirectional Transformers for Language Understanding 1 论文解读 1.1 模型概览 There are two steps in our framework: pre-training and fine-tuning. bert由预训练模型微调模型组成。 ① pre-training, the model is trained on unlabele…

前端Vue入门-day01

(创作不易,感谢有你,你的支持,就是我前行的最大动力,如果看完对你有帮助,请留下您的足迹) 目录 Vue 快速上手 Vue 概念 创建实例 插值表达式 响应式特性 开发者工具 Vue 指令 v-show v-if …

【Spring Boot】第一个Spring Boot项目:helloworld

第一个Spring Boot项目:helloworld 本节从简单的helloworld程序开始介绍创建Spring Boot项目的方法和流程,以及Spring Boot项目结构,最后介绍项目中非常重要的pom.xml文件。 1.创建Spring Boot项目 有两种方式来构建Spring Boot项目的基础…

【1++的C++初阶】之string

👍作者主页:进击的1 🤩 专栏链接:【1的C初阶】 文章目录 一,浅谈string类二,string 类常用接口2.1 string的构造2.2 string类对象的容量操作2.3 string类对象的访问及遍历操作2.4 string类对象的修改操作2.…

Python 有趣的模块之pynupt——通过pynput控制鼠标和键盘

Python 有趣的模块之pynupt ——通过pynput控制鼠标和键盘 文章目录 Python 有趣的模块之pynupt ——通过pynput控制鼠标和键盘1️⃣简介2️⃣鼠标控制与移动3️⃣键盘控制与输入4️⃣结语📢 1️⃣简介 🚀🚀🚀学会控制鼠标和键盘是…

Mongodb连接数据库

1.初始化 npm init 2.安装mongoose npm i mongoose 3.导入mongoose const mongooserequire("mongoose") 4.连接mongodb服务 mongoose.connect("mongodb://127.0.0.1:27017/user") 说明:mongodb是协议,user是数据库,如果没有会自动创…

经OPA运放后,读取电压出错

问题: 在焊接完两块板子上传程序测试时,程序上传完成,有一块板子在使用OPA读取电压时,在未插入电阻情况下,电压读取是正确的,在插入50K电压后,电压值应该是之前的两倍,但是电压变化…

unittest单元测试2

目录 unittest框架解析 构建测试套件 用例的执行顺序 unittest断言 HTML报告生成 异常捕捉与错误截图 数据驱动 🎁更多干货 完整版文档下载方式: unittest框架解析 unittest 是python 的单元测试框架,unittest 单元测试提供了创建测…

怎么把CAJ转换成PDF文件格式?分享这两个方法!

随着互联网的发展,中国知网(CNKI)已成为许多学术研究人员和学生们获取文献资料的重要来源。在CNKI上,常见的文件格式是CAJ(China Academic Journals)。然而,由于个人喜好或特定需求,我们有时会希望将这些CA…

PDF文档转化为HTML网页格式怎么操作?分享这三个方法给大家!

PDF文档作为一种常见的文档格式,广泛应用于各个领域。然而,如果您想将PDF文档直接发布到网站上,或是想在网页上进行展示,您可能需要将PDF转化为HTML格式。在此,我为大家介绍三种将PDF转化为HTML格式的方法。 方法一&am…

mysql语句练习题,创建表create ,枚举中文字符集设置,修改(update)

作业: 1.创建表: 创建员工表employee,字段如下: id(员工编号),name(员工名字),gender(员工性别),salary(员工薪…

d3dx9_43.dll丢失怎么解决(分享三个解决方法)

d3dx9_43.dll是一个Microsoft DirectX的动态链接库文件,它包含了一系列用于图形、音频和输入的功能和接口。它是DirectX 9的一部分,用于提供游戏和其他图形应用程序所需的图形和声音效果。如果计算机中d3dx9_43.dll丢失,会造成很多游戏无法打…

opencv图片根据规则改变颜色

解析 1. 读入图片 2.通道分离 3.像素值在【100,200】之间,赋值128。大于200赋值255,小于100赋值0。 源码 import cv2 img_raw_path"past/unet-test_result0-0-1-0.png" img_rawcv2.imread(img_raw_path) (r,g,b)cv2.split(img_…

运动控制介绍

运动控制介绍 1 介绍1.1 概述1.2 运动控制的基本架构1.3 常见的控制功能1.4 运动控制研究的问题分类位置变化问题周期式旋转速度变化问题 1.5 知识体系1.6 路径规划 和 轨迹规划区别与联系1.7 运动控制系统 2 《运动控制系统》[班华 李长友 主编] 摘要1 绪论1.1 运动控制研究的…

信息系统项目管理师(第四版)教材精读思维导图-第二章信息技术发展

请参阅我的另一篇文章,综合介绍软考高项: 信息系统项目管理师(软考高项)备考总结_计算机技术与软件专业技术_铭记北宸的博客-CSDN博客 思维导图源文件下载: https://download.csdn.net/download/hanjingjava/88023847 …

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试 在开发 SpringBoot 应用程序时,我们通常需要与数据库进行交互。为了确保我们的应用程序在生产环境中可以正常工作,我们需要进行数据库集成测试,以测试我们的应用程序是否能…