數值計算算法-多項式插值算法的實現與分析

數值計算是指在數值分析領域中的算法。數值分析是專門研究和數字以及近似值相關的數據問題,數值計算在數值分析的研究中發揮了特別重要的作用。

多項式插值是計算函數近似值的一種方法。其中函數值僅在幾個點上已知。

該算法的基礎是建立級數小于等于n的一個插值多項式pn(z),其中n+1是已知函數值的點的個數。

多項式插值法

許多問題都可以按照函數的方式來描述。然而,通常這個函數是未知的,我們只能通過少量的已知點來推斷函數的大致模型。為了實現這個目的,在已知點之間做插值處理。如圖所示,關于函數f(x),已知的點為x0...x8,在圖中以黑色的圓點表示。通過插值法的幫助,我們能夠獲取函數在z0、z1、z2處的值,在圖中以白色小方塊表示。本節主要討論多項式插值法。

多項式插值法的根本點就是要建立一個特殊形式的多項式,稱為插值多項式

為了深入理解插值多項式的意義,我們先來看看多項式的一些基本法則:

首先,多項式是具有如下形式的函數:

p(x) = a0 + a1x + a2x2 + ... + anxn

這里的a0,...,an是系數當an為非零整數時,這種形式的多項式稱為n階多項式。這是多項式的指數形式,在數學問題中尤為常見。但是,在某些特定的環境中其他形式的多項式則更為簡便。比如,在有關多項式插值問題中,牛頓插值多項式就是一個很好的例子

p(x) = a0 + a1 (x - c1) + a2 (x - c1)(x - c2) + ... + an(x - c1)(x - c2)...(x - cn)

這里a0,...,an系數,而c0,...,cn中值。注意到,當c0,...,cn全為0時,牛頓插值多項式就退化為前面定義的n階多項式。

構建插值多項式

下面我們來看看如何對函數f(x)構建一個插值多項式。

為了對函數f(x)進行插值,要構建一個階數小于等于n的多項式pn(z),而這又需要用到函數f(x)的n+1個已知點:x0,...,xn這些已知點x0,...,xn就稱為插值點。通過插值多項式pn(z)可以計算出函數f(x)在x=z處的近似值。插值法需要滿足點z在[x0,xn]內。可以采用如下公式來構建插值多項式pn(z)。

pn(z) = f[x0] + f[x0,x1](z-x0) + f[x0,x1,x2](z-x0)(z-x1) + ... + f[x0,...,xn](z-x0)(z-x1)...(z-xn-1)

其中函數f(x)在點x0,...,xn處的值已知,而f[x0],...,f[x0,...,xn]則稱為差商

差商可以通過點x0,...,xn以及函數f(x)在這些點處的值來計算得出。這就是牛頓插值多項式的計算公式。注意到該公式同牛頓插值多項式的相同點。差商的計算公式為:

f[xi,...,xj] = f(xi) ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 如果i=j

f[xi,...,xj] = ( f[xi+1,...xj] - f[xi,...xj-1] ) / (xj - xi) 如果i < j

?通過這個公式不難看出,當i < j時,必須預先計算出其他的差商值。例如,要計算f[x0,x1,x2,x3],就需要先計算出f[x1,x2,x3]和f[x0,x1,x2]的值。幸運的是,可以通過一個差商表來幫助我們以一種系統的方式來計算差商值。如下圖。

差商表由多行組成。最頂行保存已知點x0,...,xn 的值。第二行保存f[x0],...,f[xn]的值。要計算出表中其他的差商值,從每個待求的差商值處畫一條對角線,使其回到f[xi]和f[xj](如下圖中差商f[x1,x2,x3]處的虛線)要得到分母中的xi和xj,直接通過xi和xj求得。分子中的兩個差商就是前一階段計算出來的結果

當完成了整個差商表的計算后,插值多項式的系數就是從第二行開始,每行最左邊的那一項

計算插值多項式

一旦確定了插值多項式的系數,對于函數f,如果我們想知道某個點處的函數值,只需要對多項式求值即可

比如,已知函數f在4個點處的函數值:x0=-3.0,f(x0)=-5.0;x1=-2.0,f(x1)=-1.1;x2=2.0,f(x2)=1.9;x3=3.0,f(x3)=4.8;現在要求出點z0=-2.5,z1=0.0,z2=1.0,z3=2.5處的函數值。由于已經知道函數f在4個點處的值,因此插值多項式為3階。下圖是3階插值多項式p3(z)的差商表。

一旦從差商表中得到了系數,就可以采用前面介紹過的牛頓公式來構建插值多項式p3(z)

p3(z)=-5.0 + 3.9(z+3.0) + (-0.63)(z+3.0)(z+2.0) + 0.1767(z+3.0)(z+2.0)(z-2.0)

下一步可以通過該多項式計算出每個點z處的函數值。比如,在點z=-2.5處,經過如下計算得到

p3(z)=-5.0 + 3.9(-2.5+3.0) + (-0.63)(-2.5+3.0)(-2.5+2.0) + 0.1767(-2.5+3.0)(-2.5+2.0)(-2.5-2.0) = -2.694

點z1、z2、z3處的函數值可以通過相似的方法計算得出。最終結果以表格和函數圖像的方式表達。如下圖。

和任何其他近似算法一樣,通常會有一些與插值多項式相關的誤差出現,理解這一點很重要。定性的來講,如果要使誤差降至最小,構建的插值多項式必須要在函數f(x)上獲取足夠多的已知點才行。并且點與點之前的距離要適當,這樣最終得到的多項式才能精確地表示出函數的特性。

多項式插值的接口定義

interpol


int interpol ( const double *x, const double *fx, int n, double *z, double *pz, int m );

返回值:如果插值操作成功,返回0;否則返回-1;

描述:采用多項式插值法來求得函數在某些特定點上的值。

由調用者在參數x處指定函數值已知的點集。每個已知點所對應的函數值都在fx中指定。對應待求的點由參數z來指定,而z所對應的函數值將在pz中返回x和f(x)中的元素個數由參數n來表示。z中的待求點的個數(以及pz中返回值個數)由參數m來表示。由調用者管理x、fx、z以及pz所關聯的存儲空間。

復雜度:O(mn2),這里m代表待求值的個數,而n代表已知點的個數。

多項式插值的實現與分析

多項式插值法主要基于對一系列期望點的插值多項式的確定。要得到這個多項式,首先必須通過計算差商來確定該多項式的系數。

首先,為差商以及待確定的系數分配存儲空間。注意,由于差商表中每一行的每一項都僅依賴于其前一行的計算結果,因此,并不需要一次性保留所有的表項。所以,只為最占用空間的行分配空間即可該行將有n個條目

接下來,用fx中的值來初始化差商表的第一行。這是為計算差商表中的第三行做準備。(前兩行不需要計算,因為這兩行中的條目都已經保存在x和fx中)。

初始化的最后一步是在coeff[0]中保存fx[0]的值,因為這是插值多項式的第一個系數

計算差商的過程涉及一個嵌套循環,我們在循環中根據前面介紹過的公式來計算差商。在外層循環中,k用來統計正在計算的是哪一行(排除x和fx所代表的行)。在內層循環中i表示在當前行中正在計算的是哪一個條目一旦計算完一行的條目,table[0]中的值就成為插值多項式的下一個系數。因此,保存該值到coeff[k]中一旦得到插值多項式的所有系數,就可以計算出z中每個目標點的值,最后將這些值保存在pz中。

?我們命名這個函數為interpol,它的時間復雜度為O(mn2這里m代表z中的元素個數(也是pz中值的個數),n代表x中(也是fx中)的元素個數。復雜度因子n2是這樣得到的,變更j控制循環中的每次迭代,當前迭代中需要乘以的因子比上一輪要多一個。也就是說,當j=1時,term需要做1次乘法,當j=2時,term需要做2次乘法,持續這個過程直到j=n-1時,term需要做n-1次乘法。實際上,這就成了對1~n-1的整數求和,得到的計算時間為T(n)=(n(n+1)/2)-n,再乘以某段固定的時間。(這是由計算等差數列的著名公式得到的)。在大O記法中可以簡化為O(n2)。O(mn2)中的因子m來源于針對z中的每個點計算多項式插值的過程。在第一個嵌套循環中,計算出所有的差商,其復雜度為O(n2)。因此,最終的復雜度有一個額外的因子m,該因子對實際的復雜度沒有多大的影響。

示例:多項式插值的實現

/*interpol.c*/
#include <stdlib.h>
#include <string.h>#include "nummeths.h"/*interpol  */
int interpol(const double *x, const double *fx, int n, double *z, double *pz, int m)
{double term, *table, *coeff;int i,j,k;/*為差商和待確定的系數分配空間*/if((table = (double *)malloc(sizeof(double)*n)) == NULL)return -1;if((coeff = (double *)malloc(sizeof(double)*n)) == NULL){free(table);return -1;}/*初始化差商表*/memcpy(table,fx,sizeof(double)*n);/*重點:確定差商表的系數*/coeff[0] = table[0];for(k=1; k<n; k++)/*外層循環k用來統計正在計算的是哪一行*/{for(i=0; i<n-k; i++)/*內層循環i表示在當前行中正在計算的是哪一個條目(隨之行數的增加,條目減少)*/{j=i+k;/*當前行的每一項中分子的差商就是其前一階段計算的結果*/table[i] = (table[i+1] - table[i]) / (x[j] - x[i]);}/*當前行的首個條目計算結果即是多項式的下一個系數*/coeff[k]=table[0];}free(table);/*在指定點上對插值多項式進行求值(循環構造插值多項式)*/for(k=0; k<m; k++)          /*最外層:遍歷z點數組*/{/*插值多項式的第一個因子*/pz[k] = coeff[0];       for(j=1; j<n; j++)      /*嵌套:構造多項式(新算式等于上一步的結果加上新因子)*/{term = coeff[j];    /*因子構成:以多項式系數為基礎*/for(i=0; i<j; i++)  /*嵌套:新因子以上一步的結果乘以(z[k] - x[i]*/term=term*(z[k] - x[i]);pz[k]=pz[k] + term;}}free(coeff);return 0;
}

?

轉載于:https://www.cnblogs.com/idreamo/p/9039000.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/452648.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/452648.shtml
英文地址,請注明出處:http://en.pswp.cn/news/452648.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

HIVE ORC 報錯ClassCastException

HIVE ORC格式的表查詢報錯 Failed with exception java.io.IOException:java.lang.ClassCastException: org.apache.hadoop.hive.ql.io.orc.OrcStruct cannot be cast to org.apache.hadoop.io.BinaryComparable 建表語句如下&#xff1a; CREATE EXTERNAL TABLE test_orc( te…

程序型語言VS.編譯型語言

導讀&#xff1a;每日[快訊精選]是由CSDN研發頻道推出的特色欄目&#xff0c;每一天我們將從國外技術媒體(例如Hacker News、Reddit...等等)中挑選出有價值的新聞簡訊&#xff0c;讓您在第一時間掌握業界主流的技術文摘&#xff0c;每天清晨為您獻上第一份技術早餐。 [1]程序型…

ancestral 箭頭符號,譯林版《牛津高中英語》模塊五 高二上學期

《牛津英語》由譯林出版社和牛津大學出版社聯合編寫出版。通過在南京和蘇州開始的試用&#xff0c;取得了非常良好的效果&#xff0c;己在省內全面推廣。有人認為新教材在教育觀念和編排體系上的改革力度是八十年代以來最大的一次。它帶給我們一線教師的沖擊無疑是巨大的。二、…

[NOI2012]騎行川藏

題解&#xff1a; 我發現拉格朗日乘數法真是個好東西。。 我是不會說我數學競賽求最值都是用這個東西的 由于我不太會打那個符號就用li代表通常偏導數中的lanmuda 。。。 這題里化簡一下就可以得到 2 li * ki * ?(vi??vi′?)* vi^2?1 然后一旦li確定 我們會發現這個三次函…

MAC地址和IP地址的關系

簡單地說&#xff1a;ip地址是服務商給你的&#xff0c;mac地址是你的網卡物理地址。 一、IP地址 對于IP地址&#xff0c;相信大家都很熟悉&#xff0c;即指使用TCP/IP協議指定給主機的32位地址。IP地址由用點分隔開的4個8八位組構成&#xff0c;如192.168.0.1就是一個IP地址…

Linux中斷 - tasklet

一、前言 對于中斷處理而言&#xff0c;linux將其分成了兩個部分&#xff0c;一個叫做中斷handler&#xff08;top half&#xff09;&#xff0c;屬于不那么緊急需要處理的事情被推遲執行&#xff0c;我們稱之deferable task&#xff0c;或者叫做bottom half&#xff0c;。具體…

數字電視制播設備間的文件交換格式

在現今的數字電視演播室中&#xff0c;設備之間基本上采用信號流連接方式&#xff0c;如SDI、STDI、模擬YUV、VBS等信號流。在非線性編輯系統和播出系統與服務器之間的連接&#xff0c;還有基于MPEG-2傳輸流等的信號連接方式。基于信號流連接方式的主要特點是&#xff0c;傳送時…

oracle 位移運算符,Oracle“(+)”運算符

在Oracle中&#xff0c;()表示JOIN中的“可選”表。 所以在你的查詢中&#xff0c;select a.id, b.id, a.col_2, b.col_2, ... from a,b where a.idb.id()這是一個左外加B表與一個表。 就像現代的左連接查詢一樣。 (它將返回a表的所有數據&#xff0c;而不會丟失在另一邊的數據…

JAVA-數據類型-復習

JAVA-數據類型-復習 Java中&#xff0c;一共有8種數據類型&#xff0c;4種整型&#xff0c;2種浮點型&#xff0c;1種用于表示Unicode編碼的字符單元的字符類型char&#xff0c;1種布爾類型。 整型 類型存儲需求&#xff08;字節&#xff09;一個字節包含8個位取值范圍byte1-12…

什么是實體-聯系圖(ER圖)

實體-聯系圖&#xff08;ER圖&#xff09;數據模型中包含3種相互關聯的信息&#xff1a;數據對象、數據對象的屬性及數據對象彼此間相互連接的關系。 1.數據對象 數據對象是對軟件必須理解的復合信息的抽象。所謂符合信息是指具有一系列不同性質或屬性的事物&#xff0c;僅有單…

記錄的習慣

記錄的習慣 書籍是人類進步的階梯&#xff0c;承載了人類文明進步的歷程。大多數人都寫過日記&#xff0c;但不知道有多少人重視過日記。常常我們會用相機記錄一些生活中的場景&#xff0c;然后收藏起來&#xff0c;等到若干年后再拿出來看&#xff0c;總能感覺到很溫馨很美好。…

php 去掉實體,用PHP刪除除5個預定義HTML實體之外的所有實體的最佳方法-用于XHTML5輸出...

我目前正在嘗試提供XHTML5.目前,我在正在處理的頁面上提供XHTML 1.1 Strict.那就是我為有能力的瀏覽器所做的.對于那些不接受XML編碼數據的人,我會嚴格遵循HTML4.1.在嘗試使用HTML5進行試驗時,以HTML5格式交付時,所有功能或多或少都可以按預期工作.但是,作為XHTML5交付時,我遇到…

Flask愛家租房--發布新房源(保存房屋基本信息)

0.頁面展示效果 1.后端代碼 api.route("/houses/info", methods["POST"]) login_required def save_house_info():"""保存房屋的基本信息前端發送過來的json數據{"title":"","price":"","ar…

今后最有前途的媒體格式 MXF

MXF格式已經被推出幾年了&#xff0c;從當初一個陌生的不為人們重視的格式逐漸獲得了業內人士的認知和認可&#xff0c;現如今正被廣泛應用于廣播電視與后期制作領域&#xff0c;且有不斷擴大之勢&#xff0c;松下公司推出的基于PII卡的無磁帶式標清攝像機&#xff0c;它所采用…

【c#】RabbitMQ學習文檔(一)Hello World

一、簡介 RabbitMQ是一個消息的代理器&#xff0c;用于接收和發送消息&#xff0c;你可以這樣想&#xff0c;他就是一個郵局&#xff0c;當您把需要寄送的郵件投遞到郵筒之時&#xff0c;你可以確定的是郵遞員先生肯定會把郵件發送到需要接收郵件的人的手里&#xff0c;不…

什么是狀態轉換圖

通過描繪系統的狀態及引起系統狀態轉換的事件&#xff0c;來表示系統的行為。此外狀態轉換圖還指明了作為特定事件的結果系統將做哪些動作&#xff08;例如&#xff0c;處理數據&#xff09;。因此狀態轉換圖提供了行為建模機制。

Python學習筆記三

參考教程&#xff1a;廖雪峰官網https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000 一、函數的定義 Python中定義一個函數需要使用def語句&#xff0c;依次確定函數名、參數及函數體內容&#xff1a; #一個求絕對值的函數 def my_abs(x):i…

oracle中如何分頁,Oracle中操作分頁

mysql中分頁的寫法&#xff1a;select t.* from tbl_user t order by t.id limit $offset , $perpage$currentPage 1;//當前頁碼其中后面$sql&#xff1a;with partdata as (select rownum rowno,t.* from tablename t where column1090order by column) select * from partda…