C++实现二阶 三阶贝塞尔曲线

C++实现贝塞尔曲线

已经存在许多讲解得比较清楚的解释。如https://www.jianshu.com/p/0c9b4b681724

贝塞尔曲线包含多阶。贝塞尔曲线的阶数和次数是一样的,二阶贝塞尔,三个点,最高次数二次。例:二阶贝塞尔:三个点,两个线段,以所有等比的点组合成的曲线叫做二阶贝塞尔曲线。如下图。
在这里插入图片描述

在这里插入图片描述

根据贝塞尔曲线的公式直接根据控制点缺点阶数,。
在这里插入图片描述
定义.h头文件

class Bezier {
public:
  Bezier() = default;
  const std::vector<hybrid_astar::Vector2D>& CalculateSpline(
      const std::vector<hybrid_astar::Vector2D>& control_points,
      const int& points_number);
  void CalculateSecond();
  void CalculateCubic();
private:
  int ctrl_num_;  //控制点数量
  int order_;     //阶数,表示曲线为几阶曲线,为控制点数-1
  int knot_num_;  //节点数是分布在0-1之间有多少个点
  std::vector<hybrid_astar::Vector2D> control_points_;
  std::vector<hybrid_astar::Vector2D> path_points_;

};

定义cpp文件

const std::vector<hybrid_astar::Vector2D>& Bezier::CalculateSpline(
    const std::vector<hybrid_astar::Vector2D>& control_points,
    const int& knot_number) {
  ctrl_num_ = control_points.size();
  control_points_ = control_points;
  knot_num_ = knot_number;
  //点数-1为阶数l
  order_ = control_points_.size() - 1;
  switch (order_) {
    case 2:
      CalculateSecond();
      break;
    case 3:
      CalculateCubic();
      break;
    default:
      std::cout << "No define order: " << order_ << std::endl;
      break;
  }
  return path_points_;
}

void Bezier::CalculateSecond() {
  if (ctrl_num_ != 3)
    return;
  path_points_.clear();
  double t = 0;
  double delta_t = 1.0 / (static_cast<double>(knot_num_));
  double f1, f2, f3;
  for (double i = 0; i <= knot_num_; ++i) {
    t = delta_t * i;
    f1 = (1 - t) * (1 - t);
    f2 = 2 * (1 - t) * t;
    f3 = t * t;
    path_points_.push_back(
          hybrid_astar::Vector2D(f1 * control_points_[0].getX() +
          f2 * control_points_[1].getX() + f3 * control_points_[2].getX(),
        f1 * control_points_[0].getY() + f2 * control_points_[1].getY() +
        f3 * control_points_[2].getY()));
  }
}

void Bezier::CalculateCubic() {
  if (ctrl_num_ != 4)
    return;
  path_points_.clear();
  double t = 0;
  double delta_t = 1.0 / (static_cast<double>(knot_num_));
  double f1, f2, f3, f4;
  for (double i = 0; i <= knot_num_; ++i) {
    t = delta_t * i;
    f1 = (1 - t) * (1 - t) * (1 - t);
    f2 = 3 * (1 - t) * (1 - t) * t;
    f3 = 3 * (1 - t) * t * t;
    f4 = t * t * t;
    path_points_.push_back(
          hybrid_astar::Vector2D(f1 * control_points_[0].getX() +
          f2 * control_points_[1].getX() + f3 * control_points_[2].getX() +
        f4 * control_points_[3].getX(),
        f1 * control_points_[0].getY() + f2 * control_points_[1].getY() +
        f3 * control_points_[2].getY() + f4 * control_points_[3].getY()));
  }
}
  • 2
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
以下是用 C 代码计算三阶贝塞尔曲线长度的示例: ```c #include <stdio.h> #include <math.h> #define PRECISION 0.001 // 精度 #define STEPS 1000 // 步数 typedef struct { double x; double y; } point; double distance(point a, point b) { double dx = a.x - b.x; double dy = a.y - b.y; return sqrt(dx * dx + dy * dy); } point bezier(point p0, point p1, point p2, point p3, double t) { double u = 1 - t; double tt = t * t; double uu = u * u; double uuu = uu * u; double ttt = tt * t; point p; p.x = uuu * p0.x + 3 * uu * t * p1.x + 3 * u * tt * p2.x + ttt * p3.x; p.y = uuu * p0.y + 3 * uu * t * p1.y + 3 * u * tt * p2.y + ttt * p3.y; return p; } double bezier_length(point p0, point p1, point p2, point p3) { double length = 0; point last_point = p0; for (int i = 1; i <= STEPS; i++) { double t = (double) i / STEPS; point current_point = bezier(p0, p1, p2, p3, t); length += distance(last_point, current_point); last_point = current_point; } return length; } int main() { point p0 = {0, 0}; point p1 = {1, 2}; point p2 = {2, 3}; point p3 = {3, 1}; double length = bezier_length(p0, p1, p2, p3); printf("Length of Bezier curve: %f\n", length); return 0; } ``` 这个示例使用了基本的三阶贝塞尔曲线公式,以及分段线性近似来计算曲线长度。具体来说,它将曲线分成许多小线段,然后计算每个线段的长度,并将它们加起来得到曲线的总长度。在这个过程中,它使用了一个预定义的精度值来控制每个线段的长度,以及一个预定义的步数来控制曲线的分段数量。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值