[关闭]
@cxm-2016 2016-11-03T10:28:44.000000Z 字数 3147 阅读 3753

图形算法:贝塞尔曲线

算法

版本:1
作者:陈小默
声明:禁止商用,禁止转载

发布于:作业部落CSDN博客


一、什么是贝塞尔曲线

贝塞尔曲线是依据任意位置的点坐标绘制出的一条光滑曲线。其设计思路是按照规律u从0到1的移动过程中,在各个控制点连线的相应位置取点,并对相邻两条线上的点再次连线,重复以上过程使得没有可连接的两个点。
图3.1.1-1
对于 图3.1.1-1 点J就是由A、B、C和D四个点所控制的贝塞尔曲线的在规律u时的轨迹点。u从0变化到1时示例如下:
图3.1.1-2

二、贝塞尔曲线的数学推导过程

这里我们以二阶,也就是由三个坐标点控制的曲线为例,参考下图:
3.1.2-1
设A点坐标为 ,B点坐标为 ,C点坐标为 ,D点坐标为 ,E点坐标为 ,F点坐标为

以此类推

通过连接D、E两点得到线段DE,求点F的x轴坐标

其中的与坐标轴无关项被称为贝塞尔多项式函数(贝塞尔方程)

对于式(3.1.2.5)可以使用式(3.1.2.6)扩展为任意阶数的贝塞尔公式

假如我们使用的是三位曲线,还可以扩展Z坐标轴

三、使用C\C++语言实现贝塞尔曲线的计算过程

思路:由于贝塞尔曲线由控制点控制,在计算完成前我们无法确切的知道曲线路径上将要绘制的像素点个数,所以我们可以采取逼近的策略,即将计算出的相邻两点使用直线连接,已达到近似效果。以下过程使用C语言实现

  1. //该类用来存储三维坐标
  2. typedef struct{
  3. float x,y,z;
  4. }point3D;
  5. //整型数组
  6. typedef struct{
  7. int length;
  8. int * arr;
  9. }intArray;
  10. //坐标数组
  11. typedef struct{
  12. int length;
  13. point3D * arr;
  14. }pointArray;
  15. //该方法接受经过计算的贝塞尔点,并对相邻两个点进行连线处理
  16. void drawLine(pointArray &bezPts);
  17. //此方法用来计算二项式系数
  18. //接受一个数组对象的引用
  19. void binomialCoefficient(intArray &C){
  20. int j,n=C.length-1;
  21. for(int k=0;k<=n;k++){
  22. C.arr[k] = 1;
  23. for(j=n;j>k;j--)
  24. C.arr[k] *= j;//分子累乘
  25. for(j=n-k;j>1;j--)
  26. C.arr[k] /= j;//分母累除
  27. }
  28. }
  29. //此方法计算在第规律u处点的坐标位置
  30. //接受的第一个参数为规律u,第二个参数为将要存放位置点的指针,第三个参数为所有控制点坐标数组,第五个参数为系数数组)
  31. void computeBezPt(float u,point3D &bezPt,pointArray &ctrlPts,intArray &C){
  32. int n=ctrlPts.length-1;
  33. float bezFcn;//保存贝塞尔方程计算结果
  34. bezPt.x=bezPt.y=bezPt.z=0;//初始化
  35. for(int k=0;k<=n;k++){
  36. bezFcn = C.arr[k]*pow(u,k)*pow(1-u,n-k);
  37. bezPt.x+=ctrlPts.arr[k].x*bezFcn;
  38. bezPt.y+=ctrlPts.arr[k].y*bezFcn;
  39. bezPt.z+=ctrlPts.arr[k].z*bezFcn;
  40. }
  41. }
  42. //此方法为调用方法,用来产生贝塞尔曲线
  43. //第一个参数用来接受控制点数组,第二个参数为精度,精度越高两个点之间的距离越小,但是需要的计算时间也就越长
  44. void bezier(pointArray &ctrlPts,int precision){
  45. intArray C;
  46. C.length=ctrlPts.length;
  47. C.arr = new int[C.length];
  48. binomialCoefficient(C);//计算系数
  49. pointArray bezPts;//保存计算点的参数
  50. bezPts.length=precision+1;
  51. bezPts.arr = new point3D[bezPts.length];
  52. float u;//规律参数
  53. for(int k=0;k<=precision;k++){
  54. u = float(k)/float(precision);
  55. computeBezPt(u,bezPts.arr[k],ctrlPts,C);
  56. }
  57. drawLine(bezPts);
  58. delete[] bezPts.arr;
  59. delete[] C.arr;
  60. }

以上就是贝塞尔曲线的实现过程,这里的绘制直线函数没有定义,我们可以使用其他的图形软件包去实现连线过程。下面展示使用OpenGL的GLUT软件包的实现过程。仅供参考

  1. void drawLine(pointArray &bezPts){
  2. glBegin(GL_LINE_STRIP);
  3. for(int i=0;i<bezPts.length;i++){
  4. glVertex3f(bezPts.arr[i].x,bezPts.arr[i].y,bezPts.arr[i].z);
  5. }
  6. glEnd();
  7. }
  8. static pointArray ctrlPts_;
  9. void displayFcn(void){
  10. glClear(GL_COLOR_BUFFER_BIT);
  11. glColor3f(1,0,0);
  12. bezier(ctrlPts_,100);
  13. glFlush();
  14. }
  15. int main(int argc, char* argv[]){
  16. ctrlPts_.length = 4;
  17. point3D ctrlPts[4] = {{-40,-40,0},{-10,200,0},{10,-200,0},{40,40,0}};
  18. ctrlPts_.arr= ctrlPts;
  19. glutInit(&argc,argv);
  20. glutInitDisplayMode(GLUT_SINGLE|GLUT_RGBA);
  21. glutInitWindowPosition(50,50);
  22. glutInitWindowSize(600,600);
  23. glutCreateWindow("bezier curve");
  24. glClearColor(1,1,1,0);
  25. glMatrixMode(GL_PROJECTION);
  26. gluOrtho2D(-50,50,-50,50);
  27. glutDisplayFunc(displayFcn);
  28. glutMainLoop();
  29. return 0;
  30. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注