MATLAB三次样条插值之三弯矩法
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
MATLAB三次样条插值之三弯矩法
首先说这个程序并不完善,为了实现通用(1,2,…,n)格式解题,以及为调用追赶法程序,没有针对节点数在三个以下的情况进行分类讨论。希望能有朋友给出更好的方法。
首先,通过函数sanwanj得到方程的系数矩阵,即追赶法方程的四个向量参数,接下来调
用追赶法(在intersanwj函数中),得到三次样条分段函数系数因子,然后进行多项式合并
得到分段函数的解析式,程序最后部分通过判断输入值的区间自动选择对应的分段函数并计算
改点的值。附:追赶法程序chase
%%%%%%%%%%%%%%
function [newv,w,newu,newd]=sanwj(x,y,x0,y0,y1a,y1b)ﻫ%三弯矩样
条插值ﻫ%将插值点分两次输入,x0y0单独输入ﻫ% 边值条件a的二阶导数 y1a 和b
的二阶导数y1b,这里建议将y1a和y1b换成y2a和y2b,以便于和三转角代码相区别
ﻫn=length(x);m=length(y);
if m~=nﻫerror('x or y 输入有误,再来');
endﻫv=ones(n-1,1);u=ones(n-1,1);d=zeros(n-1,1);ﻫw=2*o
nes(n+1);ﻫh0=x(1)-x0;ﻫh=zeros(n-1,1);
for k=1:n-1ﻫh(k)=x(k+1)-x(k);ﻫend
v(1)=h0/(h0+h(1));
u(1)=1-v(1);
d(1)=6*((y(2)-y(1))/h(1)-(y(1)-y0)/h0)/(h0+h(1));ﻫ%
for k=2:n-1ﻫv(k)=h(k-1)/(h(k-1)+h(k));ﻫu(k)=1-v(k);ﻫd(k)=
6*((y(k+1)-y(k))/h(k)-(y(k)-y(k-1))/h(k-1))/(h(k-1)+h(k));
end
newv=[v;1];ﻫnewu=[1;u];
d0=6*((y(1)-y0)/h0-y1a)/h0;
d(n)=6*(y1b-(y(n)-y(n-1))/h(n-1))/h(n-1);
newd=[d0;d];
%%%%%%%%%%%%
function intersanwj(x,y,x0,y0,y1a,y1b)
%三弯矩样条插值ﻫ%第一部分ﻫn=length(x);m=length(y);
if m~=nﻫerror('xory 输入有误,再来');
endﻫ%重新定义hﻫh=zeros(n,1);
h(1)=x(1)-x0;
for k=2:n
h(k)=x(k)-x(k-1);ﻫend
%sptep1调用三弯矩函数ﻫ[a,b,c,d]=sanwj(x,y,x0,y0,y1a,y1b);
% 三对角方程ﻫM=chase(a,b,c,d);
% 求插值函数ﻫfprintf('三次样条(三弯矩)插值的函数表达式\n');
syms X ;ﻫfprintf('S0--1:\n');
S(1)=collect(((1/6)*M(2)*(X-x0).^3-(1/6)*M(1)*(X-x(1)).^3+(y(1)-(M(2)*h(1).^2)/6)*(X-x0)-(y0-(M(1)*h(1).^2)/6)*(X-x(1)))/h(1));ﻫfor k=2:n
fprintf('S%d--%d:\n',k-1,k);
S(k)=collect(((1/6)*M(k+1)*(X-x(k-1)).^3-(1/6)*M(k)*(X-x(k)).^3+(y(k)-(M(k+1)*h(k).^2)/6)*(X-x(k-1))-(y(k-1)-(M(k)*h(k).^2)/6)*(X-x(k)))/h(k));
endﻫS=S.';ﻫdisp(S);ﻫfprintf('以上为样条函数(三弯矩)解析式,显示为手写如下:\n');ﻫpretty(S);
%第二部分
%是否继续运行程序ﻫmyloop=input('继续运行程序输入“1”,否则输入“0”\n');ﻫifmyloop
while myloop
xi=input('输入需要计算的点的值,并按回车键\n');
if xi>x0|xi elseﻫfprintf('输入数值不在插值范围内,请重新输入\n');ﻫxi=input('输入需要计算的点的值,并按回车键……\n'); end %确定输入的数值应该使用哪个解析式 newx=[x0;x]; [r,suoy]=min(abs(newx-xi));ﻫfprintf('输入点的插值函数值为:\n\n');fpr intf('\t'); if xi<=newx(suoy)ﻫf=subs(S(suoy-1),X,xi);ﻫelse f=subs(S(suoy),X,xi); end disp(f);ﻫmyloop=input('继续计算输入“1”,终止计算输入“0”\n'); end else return; end %%%%%%%%%%%% function[x]=chase(a,b,c,d) %追赶法解性方程组 a是下三角b是对角线c是上三角 d是常数项 %输入的a bc d 均为列向量ﻫn=length(b);ﻫu=zeros(n,1);ﻫv=zeros(n,1);ﻫx=zeros(n,1);