曲线拟合的数值计算方法实验教材.docx

上传人:b****8 文档编号:10445522 上传时间:2023-02-11 格式:DOCX 页数:25 大小:196.63KB
下载 相关 举报
曲线拟合的数值计算方法实验教材.docx_第1页
第1页 / 共25页
曲线拟合的数值计算方法实验教材.docx_第2页
第2页 / 共25页
曲线拟合的数值计算方法实验教材.docx_第3页
第3页 / 共25页
曲线拟合的数值计算方法实验教材.docx_第4页
第4页 / 共25页
曲线拟合的数值计算方法实验教材.docx_第5页
第5页 / 共25页
点击查看更多>>
下载资源
资源描述

曲线拟合的数值计算方法实验教材.docx

《曲线拟合的数值计算方法实验教材.docx》由会员分享,可在线阅读,更多相关《曲线拟合的数值计算方法实验教材.docx(25页珍藏版)》请在冰豆网上搜索。

曲线拟合的数值计算方法实验教材.docx

曲线拟合的数值计算方法实验教材

曲线拟合的数值计算方法实验

 

【摘要】实际工作中,变量间未必都有线性关系,如服药后血药浓度与时间的关系;疾病疗效与疗程长短的关系;毒物剂量与致死率的关系等常呈曲线关系。

曲线拟合(curvefitting)是指选择适当的曲线类型来拟合观测数据,并用拟合的曲线方程分析两变量间的关系。

曲线直线化是曲线拟合的重要手段之一。

对于某些非线性的资料可以通过简单的变量变换使之直线化,这样就可以按最小二乘法原理求出变换后变量的直线方程,在实际工作中常利用此直线方程绘制资料的标准工作曲线,同时根据需要可将此直线方程还原为曲线方程,实现对资料的曲线拟合。

常用的曲线拟合有最小二乘法拟合、幂函数拟合、对数函数拟合、线性插值、三次样条插值、端点约束。

关键词曲线拟合、最小二乘法拟合、幂函数拟合、对数函数拟合、线性插值、三次样条插值、端点约束

 

一、实验目的

1.掌握曲线拟合方式及其常用函数指数函数、幂函数、对数函数的拟合。

2.掌握最小二乘法、线性插值、三次样条插值、端点约束等。

3.掌握实现曲线拟合的编程技巧。

二、实验原理

1.曲线拟合

曲线拟合是平面上离散点组所表示的坐标之间的函数关系的一种数据处理方法。

用解析表达式逼近离散数据的一种方法。

在科学实验或社会活动中,通过实验或观测得到量x与y的一组数据对(Xi,Yi)(i=1,2,...m),其中各Xi是彼此不同的。

人们希望用一类与数据的背景材料规律相适应的解析表达式,y=f(x,c)来反映量x与y之间的依赖关系,即在一定意义下“最佳”地逼近或拟合已知数据。

f(x,c)常称作拟合模型,式中c=(c1,c2,…cn)是一些待定参数。

当c在f中线性出现时,称为线性模型,否则称为非线性模型。

有许多衡量拟合优度的标准,最常用的一种做法是选择参数c使得拟合模型与实际观测值在各点的残差(或离差)

的加权平方和达到最小,此时所求曲线称作在加权最小二乘意义下对数据的拟合曲线。

有许多求解拟合曲线的成功方法,对于线性模型一般通过建立和求解方程组来确定参数,从而求得拟合曲线。

至于非线性模型,则要借助求解非线性方程组或用最优化方法求得所需参数才能得到拟合曲线,有时称之为非线性最小二乘拟合。

曲线拟合:

贝塞尔曲线与路径转化时的误差。

值越大,误差越大;值越小,越精确。

 

2.最小二乘法拟合:

最小二乘法(又称最小平方法)是一种数学优化技术。

它通过最小化误差的平方和寻找数据的最佳函数匹配。

利用最小二乘法可以简便地求得未知的数据,并使得这些求得的数据与实际数据之间误差的平方和为最小。

最小二乘法还可用于曲线拟合。

其他一些优化问题也可通过最小化能量或最大化熵用最小二乘法来表达。

函数曲线为:

y=Ax+B

其中系数满足下列的正规方程:

3.幂函数拟合:

函数曲线为:

有N个点,其中横坐标是确定的。

最小二乘幂函数拟合曲线的系数A为:

 

4.对数函数拟合:

对数函数(lograrithmicfunction)的标准式形式为

b>0时,Y随X增大而增大,先快后慢;b<0时,Y随X增大而减少,先快后慢,见图12.4(c)、(d)。

当以Y和lnX绘制的散点图呈直线趋势时,可考虑采用对数函数描述Y与X之间的非线性关系,式中的b和a分别为斜率和截距。

更一般的对数函数

Y=a+bln(X+k)

式中k为一常量,往往未知。

(a)lnY=lna+bX

(b)lnY=lna-bX

(c)Y=a+blnX

(d)Y=a-blnX

 

5.线性插值:

在代数插值中,为了提高插值多项式对函数的逼近程度一般是增加节点的个数,即提高多项式的次数,但这样做往往不能达到预想的效果。

如下图所示:

 

f(x)=1/(1+x2)如果在区间[-5,5]上取7个等距节点:

xk=5*(k/3-1)(k=0,1,2,...,6),由lagrange插值公式可得到f(x)的次L7(x)。

如图所示:

L7(x)仅在区间的中部能较好的逼近函数f(x),在其它部位差异较大,而且越接近端点,逼近效果越差。

可以证明,当节点无限加密时,Ln(x)也只能在很小的范围内收敛,这一现象称为Runge现象。

它表明通过增加节点来提高逼近程度是不适宜的,因而不采用高次多项式插值。

如果我们把以上的点用直线连接起来,显然比L7(x)要更逼近f(x)。

这就是分段线性插值。

而在实际应用中通常采用分段低次插值来提高近似程度,比如可用分段线性插值或分段三次埃尔米特插值来逼近已知函数,但它们的总体光滑性较差。

为了克服这一缺点,一种全局化的分段插值方法——三次样条插值成为比较理想的工具。

 

6.三次样条插值:

有N+1个点,其中

如果存在N个三次多项式

,系数为

满足如下性质:

则成函数S(x)为三次样条函数。

 

7.端点约束:

紧压样条:

存在唯一的三次样条曲线,其一阶导数的边界条件是:

natural样条:

存在唯一的三次样条曲线,它的自由边界条件是:

外推样条:

存在唯一的三次样条曲线,其中通过对点x1和x2进行外推得到

,同时通过对点X(n-1)和X(N-2)进行外推得到

端点曲率调整:

存在唯一的三次样条曲线,其中二阶导数的边界条件

是确定的。

抛物线终结样条:

存在唯一的三次样条曲线,其中二阶在区间[X0,X1]内

,而在[Xn-1,Xn]内

 

3、实验内容

1.P2021

胡克定律指出F=kx,其中F是拉伸弹簧的拉力(单位为盎司),x为拉伸的长度(单位为英寸)。

根据下列试验数据,求解拉伸常量k的近似值。

xk

Fk

0.2

3.6

0.4

7.3

0.6

10.9

0.8

14.5

1.0

18.2

Xk

Fk

0.2

5.3

0.4

10.6

0.6

15.9

0.8

21.2

1.0

26.4

(a)(b)

 

2.P2151

洛杉矶(美国城市)郊区11月8日的温度记录入下表所示,其中共有24个数据点。

(a)根据例5.5中的处理过程(使用fmins命令),对给定的数据求解最小二乘曲线

(b)求

(c)在同一坐标系下画出这些点集和(a)得出的最小二乘曲线。

时间,p.m.

温度

时间,a.m.

温度

1

66

1

58

2

66

2

58

3

65

3

58

4

64

4

58

5

63

5

57

6

63

6

57

7

62

7

57

8

61

8

58

9

60

9

60

10

60

10

64

11

59

11

67

午夜

58

正午

68

 

3.P2291

一个轿车在时间Tk时经过的距离dk,如下表所示。

使用程序5.3,并根据一阶导数边界条件

求这些数据的三次紧压样条插值。

时间,tk

0

2

4

6

8

距离,dk

0

40

160

300

480

 

4.P2385

美国洛杉矶郊区11月9日的温度(华氏温度)如表5.10所示。

采用24小时制。

(a)求三角多项式

(b)在同一坐标系下,画出图

和24个数据点。

(c)使用本地的温度情况重新求解问题(a)和问题(b)。

时间,p.m

温度

时间,a.m

温度

1

66

1

58

2

66

2

58

3

65

3

58

4

64

4

58

5

63

5

57

6

63

6

57

7

62

7

57

8

61

8

58

9

60

9

60

10

60

10

64

11

59

11

67

午夜

58

正午

68

 

5.P2461

编写Matlab程序,生成并绘制组合贝塞尔曲线。

利用该程序生成和绘制过3个控制点集{(0,0),(1,2),(1,1),(3,0)},{(3,0),(4,-1),(5,-2),(6,1),(7,0)},{(7,0),(4,-3),(2,-1),(0,0)}的贝塞尔曲线。

 

4、实验结果及分析

1.P2021

实验描述:

由题意可知,此题需要用最小二乘法进行计算,因为已知函数的5个插点并且知道它们的x,y的值。

且函数的表达式为F=kx,所以只需用方程中

便可计算出k的数值。

定义一个动态数组

用来依次存取x和y的插值。

其中x,y的插值通过键盘手动输入并赋予给a中的元素。

然后通过相应的求和和基本运算便可以得到相应插值下的最小二乘法的函数表达式。

(其中k保留小数点后4位)

具体函数运行效果如下:

(a)请在此输入x的各插值

0.20.40.60.81.0

请在此输入y的各插值

3.67.310.914.58.2

此函数的最小二乘法曲线表达式为

Y=14.8333*x

请按任意键继续。

(b)请在此输入x的各插值

0.20.40.60.81.0

请在此输入y的各插值

5.310.615.921.226.4

此函数的最小二乘法曲线表达式为

Y=26.4667*x

请按任意键继续。

结果分析:

易得,最小二乘法多项式计算可以很好的做出较为精确的最小二乘法拟合曲线,并且程序通用性高,只要输入相应的插值便可以计算出结果,结果系数的小数点有效位同时也可以自己确定。

 

2.P2151

实验描述:

给出的最小二乘曲线表达式为:

其中变量有5个,而给出的数据点有24个,在C语言中可以用牛顿-拉夫森算法迭代计算分别得出5个变量的值,但是方法繁琐,且迭代计算量庞大,因此这里采用Matlab进行相关的计算,调用fminsearch函数,求得当5个参量都为1附近时候多项式的最小值,此时便可求出此5个参变量的值.然后继续通过Matlab,将得到的公式以及各点,用plot函数,便可以求得。

实验结果:

运行matlab结果如下:

>>fminsearch('fiveOne',[11111])

ans=

15.72251.371715.53591.276860.3579

此时的所求值便为函数的待定系数。

所以可得最小二乘曲线的表达式为:

然后进行相应的绘制图形便可以求出所要求出的结果。

结果分析:

通过最小二乘法多项式同样可以绘制出三角函数的曲线。

并且程序通用性高,只要输入相应的插值便可以计算出结果,结果系数的小数点有效位同时也可以自己确定。

 

3.P2291

实验描述:

由题意可知,由于这里涉及到了样条线的运算,计算较为复杂。

且要涉及到画图的部分,所以此处采用matlab计算较为方便快捷。

而书本上给出了一个用来计算三次紧压样条曲线的可调用函数,现在将其引用,并根据已知点得出相应的曲线。

实验结果:

代入后得出的结果如下所示:

>>X=[02468];

>>Y=[040160300480];

>>S=csfit(X,Y,0,98)

S=

0.81258.375000

-2.437513.250043.250040.0000

1.4375-1.375067.0000160.0000

-0.81257.250078.7500300.0000

由结果可知此插值为在区间[0,8]中有三个分别划分了[0,2],[2,4],[4,6],[6,8]四个区间的插点。

且多项式分别为

在matlab中通过polyval作出相应的曲线,再用plot函数便可绘制图线。

绘制后图线如下:

此时便可以直观的看到一个平滑的样条线。

 

结果分析:

通过给定的一些数据点,便可以绘制出过这些点的相应的样条线,通过观察能发现样条线的平滑度与你选择的样条线类型以及数据点的分布有一定关联。

不仅仅是紧压样条线,相关其它的线也可以用同样的方法一一给出。

 

4.P2385

实验描述:

由题意可知,题目是想求出所绘制数据点的三角多项式的逼近,三角多项式逼近的公式为:

此外,x为离散傅里叶级数,满足条件:

实验结果:

而所给的点x为[1,24]的离散数值点,所以无法直接对其作出逼近公式,需要进行尺度变换,将x点转换为:

(1)通过matlab绘制出相关的三角多项式曲线,然后同样通过matlab的绘制点,将点绘制到这个曲线之中,具体的matlab代码如下:

>>holdon;%保持图形不动,绘制新的图形入曲线中

>>X=[123456789101112131415161718192021222324];%数据点的x的取值

>>Y=[585858585757575860646768666665646363626160605958];%数据点的y的取值

>>plot(X,Y,’xk’)%绘制出数据点

(2)然后便可以画出如图所示的插值数据点。

 

结果分析:

三角多项式的曲线拟合度非常高,能很好的绘制出图像的具体形式而且曲线平滑,但是它需要满足x属于-pi到pi的区间内。

5.P2461

实验描述:

由题意可知,首先以3阶贝塞尔函数为带求函数。

其求解格式如下:

其中b为伯恩斯坦多项式,其定义如下:

用matlab先设置一个参数t,然后再根据公式,通过所给的数据点将伯恩斯坦多项式,以及x,y的关于参数t的多项式求出来。

然后再把t设置为精度足够大的单位序列,绘制图线即可得出贝塞尔的效果。

实验结果:

其matlab运行后结果如下:

【以第一组控制点为例】

>>X=[0113]

X=

0113

>>Y=[0210]

Y=

0210

>>fiveTwo(X,Y)

x=

3*t*(t-1)^2-3*t^2*(t-1)+3*t^3

y=

6*t*(t-1)^2-3*t^2*(t-1)

 

且绘制出第一组控制点所在位置以及三阶贝塞尔曲线如下所示:

同理,第2,3组控制点所作的图形如下所示:

结果分析:

可以通过Matlab函数绘制出三阶的贝塞尔函数。

只要给出控制点,便可自动绘制出所控制的三阶贝塞尔函数以及控制点的位标。

由此可以观察到贝塞尔与控制点的约束作用,以及所求得贝塞尔函数是个相对平滑的曲线。

 

5、实验结论

曲线拟合对于实际的工程以及理论推导问题都有着重大的作用。

在具体的实验,数据分析上,往往我们只有巨量的已知离散点,想从离散点中得出函数表达式则就需要曲线拟合进行,根据不同的要求,我们可以选择最小二乘法的幂函数拟合或者是一次函数,二次函数拟合,抑或是精度非常高的傅里叶变换的三角函数拟合。

同时在建模方面,贝塞尔函数的引用也从数学层面解决了如何用计算机绘制出光滑圆润的曲线,在一些设计软件中,例如3dmax和maya的三维建模,AutoCAD的工程建模,贝塞尔运算对于曲线曲面的设计有着举足轻重的作用。

 

附件(代码):

 

1.P2021

 

#include

#include

#include

voidmain()

{

externfloatafunction(floatb[]);

externfloatbfunction(floatz,floatv);

float*a,y,q,c;

inti;

a=newfloat[5];//定义动态数组a[]

printf("请在此输入x的各插值\n");

scanf("%f%f%f%f%f",&a[0],&a[1],&a[2],&a[3],&a[4]);//手动输入x各值

q=afunction(a);//对x值进行求和

a=newfloat[5];//重新为a[]进行数据分配

printf("请在此输入y的各插值\n");

scanf("%f%f%f%f%f",&a[0],&a[1],&a[2],&a[3],&a[4]);//手动输入y各值

c=afunction(a);//对y值进行求和

y=bfunction(q,c);//计算出k的系数的大小

printf("此数据的最小二乘法曲线表达式为\nY=%f*x\n",y);

return;

}

floatafunction(floatb[])

{

floatx=0;

inti;

for(i=0;i<5;i++)

x=b[i]+x;

returnx;

}

floatbfunction(floatz,floatv)

{

floatx;

x=v/z;

returnx;

}

 

2.P2151

functionz=fiveOne(u,y)

A=u

(1);

B=u

(2);

C=u(3);

D=u(4);

E=u(5);

Z=0;

fori=1:

1:

24

z=(A.*cos(i*B)+C.*sin(i*D)+E-y(i)).^2+z;%函数的线性最小二乘法的多项式

end

之后在Matlab的命令窗口输入如下语句:

>>fminsearch(‘fiveOne’,[11111])

(绘制图形方程组)

>>x1=[0123456789101112131415161718192021222324];%数值点的x取值

>>y1=[585858585757575860646768666665646363626160605958];

%数值点的x取值

>>plot(x1,y1,'x')%绘制出24个数值点的图形

>>x=linspace(0,25,100);

>>holdon

>>plot(x,y,'k')%绘制出此函数的二乘法后的函数曲线

 

3.P2291

(三次紧压样条线计算函数)

functionS=csfit(x,y,dx0,dxn)

N=length(x)-1;

H=diff(x);

D=diff(y)./H;%d(k)的值

A=H(2:

N-1);

B=2*(H(1:

N-1)+H(2:

N));

C=H(2:

N);

U=6*diff(D);%U(k)的值

B

(1)=B

(1)-H

(1)/2;

U

(1)=U

(1)-3*(D

(1)-dx0);

B(N-1)=B(N-1)-H(N)/2;

U(N-1)=U(N-1)-3*(dxn-D(N));

fork=2:

N-1

temp=A(k-1)/B(k-1);

B(k)=B(k)-temp*C(k-1);

U(k)=U(k)-temp*U(k-1);

end

M(N)=U(N-1)/B(N-1);

fork=N-2:

-1:

1

M(k+1)=(U(k)-C(k)*M(k+2))/B(k);

end

M

(1)=3*(D

(1)-dx0)/H

(1)-M

(2)/2;%三次紧压约束中S"(0)的值

M(N+1)=3*(dxn-D(N))/H(N)-M(N)/2;%三次紧压约束中S"(k)的值

fork=0:

N-1

S(k+1,1)=(M(k+2)-M(k+1))/(6*H(k+1));%S(k,3)的值

S(k+1,2)=M(k+1)/2;%S(k,2)的值

S(k+1,3)=D(k+1)-H(k+1)*(2*M(k+1)+M(k+2))/6;%S(k,1)的值

S(k+1,4)=y(k+1);%S(k,0)的值

End

(曲线绘制函数程序)

>>x1=0:

.01:

2;y1=polyval(S(1,:

),x1-X

(1));%第一段样条线

>>x2=2:

.01:

4;y2=polyval(S(2,:

),x2-X

(2));%第二段样条线

>>x3=4:

.01:

6;y3=polyval(S(3,:

),x3-X(3));%第三段样条线

>>x4=6:

.01:

8;y4=polyval(S(4,:

),x4-X(4));%第四段样条线

>>plot(x1,y1,x2,y2,x3,y3,x4,y4,X,Y,'x')%绘制连接成完整曲线,并且标注数据点的位置。

 

#include

#include

#include

voidmain()

{

externfloatafunction(floatX[],floatY[],inttemp);

externfloatbfunction(floatX[],floatY[],inttemp);

inti;

floata[24]={1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24};

floatb[24]={58,58,58,58,57,57,57,58,60,64,67,68,66,66,65,64,63,63,62,61,60,60,59,58};

float*c;

float*d;

c=newfloat[7],d=newfloat[7];

for(i=0;i<=24;i++)

{

c[i]=afunction(a,b,i);

d[i]=bfunction(a,b,i);

}

printf("由所给的数据点可以得出\n三角多项式T7(x)中a的系数分别为\n");

printf("%f%f%f%f%f%f%f\n",c[0],c[1],c[2],c[3],c[4],c[5],c[6]);

printf("由所给的数据点可以得出\n三角多项式T7(x)中b的系数分别为\n");

printf("%f%f%f%f%f%f%f\n",d[0],d[1],d[2],d[3],d[4],d[5],d[6]);

system("pause");

return;

}

floatafunction(floatX[],floatY[],inttemp)

{

floatA=0;

inti;

for(i=0;i<=24;i++)

A=Y[i]*cos(temp*X[i])/6+A;

returnA;

}

floatbfunction(floatX[],floatY[],inttemp)

{

floatA=0;

inti;

for(i=0;i<=24;i++)

A=Y[i]

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 高等教育 > 管理学

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1