最近刚看完贝塞尔曲线,工作就遇到了相应的需求,所以写一下过程。主要讲的是自动驾驶中,车换道时用到贝塞尔曲线,当然其他的很多领域也会有,例如图形学等。
在车遇到障碍物或者是前车速度较慢的时候,就会进入换道逻辑,那么如何从一个车道换到另外一个车道,同时要保证车里面的人的一个体感问题,就是如何平滑过度,这就是为什么要使用贝塞尔曲线来做换道时的轨迹生成了,OK那开始讲贝塞尔曲线了。
对于一阶贝塞尔曲线,从上图我们可以看到,它是一条直线,通过几何知识,很容易根据t 的值,得出线段上某个点的坐标:
B 1 ( t ) = P 0 + ( P 1 − P 0 ) t B_{1}(t) = P_{0} + (P_{1}-P_{0})t B1(t)=P0+(P1−P0)t
或者
B 1 ( t ) = ( 1 − t ) P 0 + P 1 t , 0 < t < 1 B_{1}(t) = (1-t)P_{0} + P_{1}t , 0
一阶贝塞尔曲线, 就是根据t来的线性插值。P0和P1表示的是一个向量[ x , y ] , 其中x、y是分别按照这个公式来计算的。
定义:起始点、终止点(也称锚点)、控制点。通过调整控制点,贝塞尔曲线的形状会发生变化。
上面的红色曲线就是通过二阶公式生成的贝塞尔曲线(是不是很平滑)。
下面是知乎大佬的原理讲解,比较好理解。
在平面内任选 3 个不共线的点,依次用线段连接。在第一条线段上任选一个点 D。计算该点到线段起点的距离 AD,与该线段总长 AB 的比例。
根据上一步得到的比例,从第二条线段上找出对应的点 E,使得 AD:AB = BE:BC。
这时候DE又是一条直线了, 就可以按照一阶的贝塞尔方程来进行线性插值了, t= AD:AE
这时候就可以推出公式了
P 0 ′ ( t ) = ( 1 − t ) P 0 + P 1 t , 0 < t < 1 P_{0}'(t) = (1-t)P_{0} + P_{1}t , 0
<==>对应着上图绿色线段的左端点。
P 1 ′ = ( 1 − t ) P 1 + P 2 t P_{1}' = (1-t)P_{1} + P_{2}t P1′=(1−t)P1+P2t
<==>对应着上图绿色线段的右端点。
B 2 ( t ) = ( 1 − t ) P 0 ′ + P 1 ′ t B_{2}(t) = (1-t)P_{0}' + P_{1}'t B2(t)=(1−t)P0′+P1′t
= ( 1 − t ) ( ( 1 − t ) P 0 + t P 1 ) + t ( ( 1 − t ) P 1 + t P 2 ) = (1-t)((1-t)P_{0}+tP_{1})+t((1-t)P_{1}+tP_{2}) =(1−t)((1−t)P0+tP1)+t((1−t)P1+tP2)
= ( 1 − t ) 2 P 0 + 2 t ( 1 − t ) P 1 + t 2 P 2 = (1-t)^2P_{0}+2t(1-t)P_{1}+t^2P_{2} =(1−t)2P0+2t(1−t)P1+t2P2
<==>对应着绿色线段的一阶贝塞尔曲线(线性插值)。
整理一下公式, 得到二阶贝塞尔公式:
B 2 ( t ) = ( 1 − t ) 2 P 0 + 2 t ( 1 − t ) P 1 + t 2 P 2 B_{2}(t) = (1-t)^2P_{0}+2t(1-t)P_{1}+t^2P_{2} B2(t)=(1−t)2P0+2t(1−t)P1+t2P2
可以通过递归的方式来理解贝塞尔曲线, 但是还是给出公式才方便计算的。划重点了: 系数是二项式的展开. 后面的很多的贝塞尔曲线的性质都可以用这个来解释
P ( t ) = ∑ i = 0 n P i B i , n ( t ) , 0 < = t < = 1 P_{}(t) = \sum_{i=0}^{n}P_{i}B_{i,n}(t),0<=t<=1 P(t)=i=0∑nPiBi,n(t),0<=t<=1
随着阶数的变化,图像中贝塞尔曲线也会跟着变化。
通过上面图,可以看出,最终的(红色)曲线,就是对这几个点进行拟合得到的贝塞尔曲线。
n个控制点对应着n-1 阶的贝塞尔曲线。
高阶的贝塞尔可以通过不停的递归直到一阶:
贝塞尔曲线始终会在包含了所有控制点的最小凸多边形中, 不是按照控制点的顺序围成的最小多边形。这点大家一定注意. 这一点的是很关键的,也就是说可以通过控制点的凸包来限制规划曲线的范围,在路径规划是很需要的一个性质.
凸包可以理解为,有一堆点集,使用一个橡皮筋来套住所有点,最后橡皮筋围成的形状,就是这些点集的凸包。上面最后一个图的5个点中,其实最后一个点P4不是在凸多边形上,而是在这些点组成的凸包内部。
用不严谨的话来讲,给定二维平面上的点集,凸包就是将最外层的点连接起来构成的凸多边形,它能包含点集中所有的点。
通过选取几个控制点来生成贝塞尔曲线,该曲线为自动驾驶车的参考线,车沿着轨迹走。
QT代码的实现:
//递归
double factorial(int n){
if(n<=1){
return 1;
}else {
return factorial(n-1)*n;
}
}
//贝塞尔公式
Vector2d bezierCommon(vector<Vector2d> Ps, double t){
if(Ps.size()==1){
return Ps[0];
}
Vector2d p_t(0.,0.);
int n = Ps.size()-1;
for (int i= 0; i< Ps.size(); i++) {
double C_n_i = factorial(n)/(factorial(i)*factorial(n-i));
p_t += C_n_i*pow((1-t),(n-i))*pow(t,i)*Ps[i];
}
return p_t;
}
//主函数调用
int MainWindow::bezier(){
vector<Vector2d>Ps{Vector2d (0,1),Vector2d(1,3),Vector2d(3,1),Vector2d(4,6),Vector2d(5,9)};
vector<double>x_ref,y_ref;
for(int i=0;i<Ps.size();i++){
x_ref.push_back(Ps[i][0]);
y_ref.push_back(Ps[i][1]);
}
for(int t=0;t<100;t++){
Vector2d pos = bezierCommon(Ps,(double)t/100);
x_.push_back(pos[0]);
y_.push_back(pos[1]);
return 0;
}
/****************************************************
*贝塞尔曲线画图
****************************************************/
int result = bezier();
QSplineSeries* series = new QSplineSeries(); // 创建一个样条曲线对象
QScatterSeries* seriesPoint = new QScatterSeries(); //散点
series->setName("曲线");
#if 1
// 添加生成的贝塞尔曲线的点
for (int var = 0; var < x_.size(); ++var) {
series->append(x_[var],y_[var]);
}
// 添加控制点
*seriesPoint << QPointF(0, 1)<< QPointF(1, 3)<< QPointF(3, 1)<< QPointF(4, 6)<<QPointF(5,9);
#else // 添加数据方式3,一次性更新所有数据
QList<QPointF> points;
for(int i = 0; i < 20; i++)
{
points.append(QPointF(i, i %7));
}
series->replace(points);
#endif
参考:
https://www.zhihu.com/question/29565629