查看: 2597|回复: 0
打印 上一主题 下一主题

[as3基础应用]三次贝塞尔曲线绘制算法(优化过)

[复制链接]
.    

3797

主题

11

听众

5万

积分

首席设计师

Rank: 8Rank: 8

纳金币
32328
精华
41

活跃会员 优秀版主 荣誉管理 论坛元老

跳转到指定楼层
楼主
发表于 2013-3-20 09:09:54 |只看该作者 |倒序浏览
动画地址:http://boycy.webs.com/bezier.swf
这学期学图形学,就把自己的一些粗浅的理解发上去让大家拍砖。前些天做三次贝塞尔曲线绘制的上机练习,正好将从直线扫描算法中启发得来的n次多项式批量计算用上了,自认为优化得还可以。原来写的版本是C++,为了便于网上观看特改写成AS3,对这方面内行的朋友,希望还多多指点!
在讲三次贝塞尔曲线之前,先给出n次贝塞尔曲线的一般式:
[R]( t ) = ( +..i=0,n ) ( [R]i * Bi,n( t ) ) , t属于[0,1]
Bi,n( t ) = Cni * pow( 1 - t , n - i ) * pow( t , i )
这里我把符号说明下,中括号[]括起来的表示矩阵。比如[R]就是个矩阵,如果我们讨论的贝塞尔曲线的点在平面内,那么矩阵里面有两个元素,比如我们熟知的x,y,如果是空间的曲线,就是三个元素,元素再多。。那就不知道是几次元了,反正是个数学概念。在计算的时候,将矩阵的元素单独拎出来计算也是可以的,因为公式中并没有出现矩阵和矩阵相乘,也就是说,各元素计算时其实各不相干。
( +..i=0,n ) 这个运算符,这个符号是我自己发明的,其实是累加符号,表示公式后面括号内将i分别用0,1,2,3..n代替,每个代替运算的结果全部加起来。
Cni是二项式系数,其值为 n! / ( i! * ( n - i )! )
pow是求指数函数。
其中 Bi,n( t ) = Cni * pow( 1 - t , n - i ) * pow( t , i ) 为n次二项式展开后的一项。其意义在后面的三次贝塞尔曲线里面讨论更容易理解
那么,现在先讨论3次贝塞尔曲线,我们把公式中的n全部替换成3,得到三次贝塞尔曲线的一般式如下:
[R]( t ) = ( +..i=0,3 ) ( [R]i * Bi,3( t ) ) , t属于[0,1]
Bi,3( t ) = C3i * pow( 1 - t , 3 - i ) * pow( t , i )
直接把Bi,3( t )代入吧,得到式子:
[R]( t ) = ( +..i=0,3 ) ( [R]i * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
对于不熟悉线性代数或者不习惯用矩阵表示数据的朋友,我们就直接把矩阵拆开好了,[R]矩阵中总共就两个元素,分别是x和y,这里就把x和y分别计算,得到关于x和y的参数曲线:
X( t ) = ( +..i=0,3 ) ( Xi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
Y( t ) = ( +..i=0,3 ) ( Yi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
如果对参数曲线不熟悉,可以参考直线的参数方程
比如一条直线经过(x1,y1),(x2,y2),那么其参数方程可表示为:
X(t)=x1+t*(x2-x1)
Y(t)=y1+t*(y2-y1)
如果是线段那么t在0和1之间,如果是直线,t是实数。
言归正传,下面我们开始处理参数曲线,由于x的参数方程和y的形式差不多,我们就以x的参数方程做例子来分析
X( t ) = ( +..i=0,3 ) ( Xi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
首先我们把连加拆开,那么i分别等于0,1,2,3,得到一个式子:
X( t ) =
X0 * C30 * pow( 1 - t , 3 - 0 ) * pow( t , 0 )
+
X1 * C31 * pow( 1 - t , 3 - 1 ) * pow( t , 1 )
+
X2 * C32 * pow( 1 - t , 3 - 2 ) * pow( t , 2 )
+
X3 * C33 * pow( 1 - t , 3 - 3 ) * pow( t , 3 )
t属于[0,1]
看这个式子,我们可以发现,C30 * pow( 1 - t , 3 - 0 ) * pow( t , 0 )+C31 * pow( 1 - t , 3 - 1 ) * pow( t , 1 )+C32 * pow( 1 - t , 3 - 2 ) * pow( t , 2 )+C33 * pow( 1 - t , 3 - 3 ) * pow( t , 3 )=pow( 1 - t + t , 3)=1
那就是说,x0,x1,x2,x3相加前的系数和为1,我们称这些系数为各点的权重。当然,点的权重是随着t的值连续变化的,这也是为什么三次贝塞尔曲线能光滑地通过四个点控制的原因。
下面我们进一步对式子进行化简。
其中C30等于1,C31等于3,C32等于3,C33等于1,把能合并的都合并,能计算值的都计算出来,将式子化简:
X( t ) =
X0 * pow( 1 - t , 3 )
+
X1 * 3 * pow( 1 - t , 2 ) * t
+
X2 * 3 * ( 1 - t ) * pow( t , 2 )
+
X3 * pow( t , 3 )
t属于[0,1]
这样的式子应该很好求了,他是一个最高次为3次的多项式,在
http://hi.baidu.com/boycy/blog/item/22f48945bee68633cffca3f2.html
中提到,a*pow(x,n)+b*pow(x,n-1)+..这样形式的式子,都能通过多项式批量计算方法来快速求得结果。
对于最高三次的式子来说,我们需要前三个原始数据,以及式子求导三次的常数,算法细节就不讲了,细节参考上面那篇博文。
代码:
package
{
       import flash.display.Bitmap;
       import flash.display.BitmapData;
       import flash.display.Shape;
       import flash.display.Sprite;
       import flash.events.Event;
       import flash.events.MouseEvent;
       import flash.geom.Point;
       import flash.geom.Rectangle;
       /**
        *贝塞尔曲线绘制算法
        * @author boycy815
        */
       public class Main extends Sprite
       {
              /*C30到C33的值总共才4个直接枚举*/
              private const C30:int = 1;
              private const C31:int = 3;
              private const C32:int = 3;
              private const C33:int = 1;
              /*定义曲线上点的个数(包括两端)*/
              private const LENGTH:int = 10000;
              /*btC30到btC33分别是R0 R1 R2 R3四个点当t取前三个值时的权重值*/
              private var btC30:Vector.<Number> = new Vector.<Number>(3);
              private var btC31:Vector.<Number> = new Vector.<Number>(3);
              private var btC32:Vector.<Number> = new Vector.<Number>(3);
              private var btC33:Vector.<Number> = new Vector.<Number>(3);
              /*画点的位图*/
              private var data:BitmapData;
              private var map:Bitmap;
              /*用于连接四个点的四条直线*/
              private var lin:Shape;
              /*分别是四个点的显示对象*/
              private var ct0:Sprite;
              private var ct1:Sprite;
              private var ct2:Sprite;
              private var ct3:Sprite;
              public function Main():void
              {
                     if (stage) init();
                     else addEventListener(Event.ADDED_TO_STAGE, init);
              }
              
              private function init(e:Event = null):void
              {
                     removeEventListener(Event.ADDED_TO_STAGE, init);
                     /*初始化位图和一些显示对象*/
                     data = new BitmapData(800, 600,false);
                     map = new Bitmap(data);
                     this.addChild(map);
                     lin = new Shape();
                     this.addChild(lin);
                     ct0 = new Sprite();
                     ct0.buttonMode = ***e;
                     ct0.graphics.lineStyle(0, 0x00000);
                     ct0.graphics.beginFill(0xffffff);
                     ct0.graphics.drawCircle(0, 0, 5);
                     ct0.graphics.endFill();
                     ct0.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct0.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct0);
                     ct1 = new Sprite();
                     ct1.buttonMode = ***e;
                     ct1.graphics.lineStyle(0, 0x00000);
                     ct1.graphics.beginFill(0xffffff);
                     ct1.graphics.drawCircle(0, 0, 5);
                     ct1.graphics.endFill();
                     ct1.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct1.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct1);
                     ct2 = new Sprite();
                     ct2.buttonMode = ***e;
                     ct2.graphics.lineStyle(0, 0x00000);
                     ct2.graphics.beginFill(0xffffff);
                     ct2.graphics.drawCircle(0, 0, 5);
                     ct2.graphics.endFill();
                     ct2.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct2.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct2);
                     ct3 = new Sprite();
                     ct3.buttonMode = ***e;
                     ct3.graphics.lineStyle(0, 0x00000);
                     ct3.graphics.beginFill(0xffffff);
                     ct3.graphics.drawCircle(0, 0, 5);
                     ct3.graphics.endFill();
                     ct3.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct3.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct3);
【来源:互联网】
更多精彩教程,尽在web3D纳金网http://www.narkii.com/college/ 动画地址:http://boycy.webs.com/bezier.swf
这学期学图形学,就把自己的一些粗浅的理解发上去让大家拍砖。前些天做三次贝塞尔曲线绘制的上机练习,正好将从直线扫描算法中启发得来的n次多项式批量计算用上了,自认为优化得还可以。原来写的版本是C++,为了便于网上观看特改写成AS3,对这方面内行的朋友,希望还多多指点!
在讲三次贝塞尔曲线之前,先给出n次贝塞尔曲线的一般式:
[R]( t ) = ( +..i=0,n ) ( [R]i * Bi,n( t ) ) , t属于[0,1]
Bi,n( t ) = Cni * pow( 1 - t , n - i ) * pow( t , i )
这里我把符号说明下,中括号[]括起来的表示矩阵。比如[R]就是个矩阵,如果我们讨论的贝塞尔曲线的点在平面内,那么矩阵里面有两个元素,比如我们熟知的x,y,如果是空间的曲线,就是三个元素,元素再多。。那就不知道是几次元了,反正是个数学概念。在计算的时候,将矩阵的元素单独拎出来计算也是可以的,因为公式中并没有出现矩阵和矩阵相乘,也就是说,各元素计算时其实各不相干。
( +..i=0,n ) 这个运算符,这个符号是我自己发明的,其实是累加符号,表示公式后面括号内将i分别用0,1,2,3..n代替,每个代替运算的结果全部加起来。
Cni是二项式系数,其值为 n! / ( i! * ( n - i )! )
pow是求指数函数。
其中 Bi,n( t ) = Cni * pow( 1 - t , n - i ) * pow( t , i ) 为n次二项式展开后的一项。其意义在后面的三次贝塞尔曲线里面讨论更容易理解
那么,现在先讨论3次贝塞尔曲线,我们把公式中的n全部替换成3,得到三次贝塞尔曲线的一般式如下:
[R]( t ) = ( +..i=0,3 ) ( [R]i * Bi,3( t ) ) , t属于[0,1]
Bi,3( t ) = C3i * pow( 1 - t , 3 - i ) * pow( t , i )
直接把Bi,3( t )代入吧,得到式子:
[R]( t ) = ( +..i=0,3 ) ( [R]i * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
对于不熟悉线性代数或者不习惯用矩阵表示数据的朋友,我们就直接把矩阵拆开好了,[R]矩阵中总共就两个元素,分别是x和y,这里就把x和y分别计算,得到关于x和y的参数曲线:
X( t ) = ( +..i=0,3 ) ( Xi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
Y( t ) = ( +..i=0,3 ) ( Yi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
如果对参数曲线不熟悉,可以参考直线的参数方程
比如一条直线经过(x1,y1),(x2,y2),那么其参数方程可表示为:
X(t)=x1+t*(x2-x1)
Y(t)=y1+t*(y2-y1)
如果是线段那么t在0和1之间,如果是直线,t是实数。
言归正传,下面我们开始处理参数曲线,由于x的参数方程和y的形式差不多,我们就以x的参数方程做例子来分析
X( t ) = ( +..i=0,3 ) ( Xi * C3i * pow( 1 - t , 3 - i ) * pow( t , i ) ) , t属于[0,1]
首先我们把连加拆开,那么i分别等于0,1,2,3,得到一个式子:
X( t ) =
X0 * C30 * pow( 1 - t , 3 - 0 ) * pow( t , 0 )
+
X1 * C31 * pow( 1 - t , 3 - 1 ) * pow( t , 1 )
+
X2 * C32 * pow( 1 - t , 3 - 2 ) * pow( t , 2 )
+
X3 * C33 * pow( 1 - t , 3 - 3 ) * pow( t , 3 )
t属于[0,1]
看这个式子,我们可以发现,C30 * pow( 1 - t , 3 - 0 ) * pow( t , 0 )+C31 * pow( 1 - t , 3 - 1 ) * pow( t , 1 )+C32 * pow( 1 - t , 3 - 2 ) * pow( t , 2 )+C33 * pow( 1 - t , 3 - 3 ) * pow( t , 3 )=pow( 1 - t + t , 3)=1
那就是说,x0,x1,x2,x3相加前的系数和为1,我们称这些系数为各点的权重。当然,点的权重是随着t的值连续变化的,这也是为什么三次贝塞尔曲线能光滑地通过四个点控制的原因。
下面我们进一步对式子进行化简。
其中C30等于1,C31等于3,C32等于3,C33等于1,把能合并的都合并,能计算值的都计算出来,将式子化简:
X( t ) =
X0 * pow( 1 - t , 3 )
+
X1 * 3 * pow( 1 - t , 2 ) * t
+
X2 * 3 * ( 1 - t ) * pow( t , 2 )
+
X3 * pow( t , 3 )
t属于[0,1]
这样的式子应该很好求了,他是一个最高次为3次的多项式,在
http://hi.baidu.com/boycy/blog/item/22f48945bee68633cffca3f2.html
中提到,a*pow(x,n)+b*pow(x,n-1)+..这样形式的式子,都能通过多项式批量计算方法来快速求得结果。
对于最高三次的式子来说,我们需要前三个原始数据,以及式子求导三次的常数,算法细节就不讲了,细节参考上面那篇博文。
代码:
package
{
       import flash.display.Bitmap;
       import flash.display.BitmapData;
       import flash.display.Shape;
       import flash.display.Sprite;
       import flash.events.Event;
       import flash.events.MouseEvent;
       import flash.geom.Point;
       import flash.geom.Rectangle;
       /**
        *贝塞尔曲线绘制算法
        * @author boycy815
        */
       public class Main extends Sprite
       {
              /*C30到C33的值总共才4个直接枚举*/
              private const C30:int = 1;
              private const C31:int = 3;
              private const C32:int = 3;
              private const C33:int = 1;
              /*定义曲线上点的个数(包括两端)*/
              private const LENGTH:int = 10000;
              /*btC30到btC33分别是R0 R1 R2 R3四个点当t取前三个值时的权重值*/
              private var btC30:Vector.<Number> = new Vector.<Number>(3);
              private var btC31:Vector.<Number> = new Vector.<Number>(3);
              private var btC32:Vector.<Number> = new Vector.<Number>(3);
              private var btC33:Vector.<Number> = new Vector.<Number>(3);
              /*画点的位图*/
              private var data:BitmapData;
              private var map:Bitmap;
              /*用于连接四个点的四条直线*/
              private var lin:Shape;
              /*分别是四个点的显示对象*/
              private var ct0:Sprite;
              private var ct1:Sprite;
              private var ct2:Sprite;
              private var ct3:Sprite;
              public function Main():void
              {
                     if (stage) init();
                     else addEventListener(Event.ADDED_TO_STAGE, init);
              }
              
              private function init(e:Event = null):void
              {
                     removeEventListener(Event.ADDED_TO_STAGE, init);
                     /*初始化位图和一些显示对象*/
                     data = new BitmapData(800, 600,false);
                     map = new Bitmap(data);
                     this.addChild(map);
                     lin = new Shape();
                     this.addChild(lin);
                     ct0 = new Sprite();
                     ct0.buttonMode = ***e;
                     ct0.graphics.lineStyle(0, 0x00000);
                     ct0.graphics.beginFill(0xffffff);
                     ct0.graphics.drawCircle(0, 0, 5);
                     ct0.graphics.endFill();
                     ct0.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct0.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct0);
                     ct1 = new Sprite();
                     ct1.buttonMode = ***e;
                     ct1.graphics.lineStyle(0, 0x00000);
                     ct1.graphics.beginFill(0xffffff);
                     ct1.graphics.drawCircle(0, 0, 5);
                     ct1.graphics.endFill();
                     ct1.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct1.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct1);
                     ct2 = new Sprite();
                     ct2.buttonMode = ***e;
                     ct2.graphics.lineStyle(0, 0x00000);
                     ct2.graphics.beginFill(0xffffff);
                     ct2.graphics.drawCircle(0, 0, 5);
                     ct2.graphics.endFill();
                     ct2.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct2.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct2);
                     ct3 = new Sprite();
                     ct3.buttonMode = ***e;
                     ct3.graphics.lineStyle(0, 0x00000);
                     ct3.graphics.beginFill(0xffffff);
                     ct3.graphics.drawCircle(0, 0, 5);
                     ct3.graphics.endFill();
                     ct3.addEventListener(MouseEvent.MOUSE_DOWN, mouseDown);
                     ct3.addEventListener(MouseEvent.MOUSE_UP, mouseUp);
                     this.addChild(ct3);
【来源:互联网】
更多精彩教程,尽在web3D纳金网http://www.narkii.com/college/
分享到: QQ好友和群QQ好友和群 腾讯微博腾讯微博 腾讯朋友腾讯朋友 微信微信
转播转播0 分享淘帖0 收藏收藏0 支持支持0 反对反对0
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

手机版|纳金网 ( 闽ICP备2021016425号-2/3

GMT+8, 2025-1-27 22:49 , Processed in 0.097582 second(s), 33 queries .

Powered by Discuz!-创意设计 X2.5

© 2008-2019 Narkii Inc.

回顶部