插值(一)——多项式插值(C++)

插值

插值的作用是可以将原本比较难计算的函数转换为误差在一定范围内的多项式,比如在单片机中直接计算 x 、 log ⁡ 2 x \sqrt{x}、\log_2x x log2x之类的函数是比较麻烦的,但是使用插值的方法就可以将其转换为误差可控的只有乘法和加减法的多项式,从而简化计算。

多项式插值

多项式插值是利用多项式来拟合一系列离散的数据点,从而达到简化计算的目的。本文主要介绍最“暴力”的插值方法。

  1. 设定所需构造的插值多项式:
    D n ( x ) = a 0 + a 1 x + a 2 x 2 + ⋯ + a n x n D_n(x)=a_0+a_1x+a_2x^2+\cdots +a_nx^n Dn(x)=a0+a1x+a2x2++anxn
  2. 由插值条件可得:
    D n ( x i ) = y i , i = 0 , 1 , ⋯ n D_n(x_i)=y_i,i=0,1,\cdots n Dn(xi)=yi,i=0,1,n
  3. 由此可以得到对应的方程组:
    { 1 a 0 + a 1 x 0 + a 2 x 0 2 + ⋯ + a n x 0 n = y 0 1 a 0 + a 1 x 1 + a 2 x 1 2 + ⋯ + a n x 1 n = y 1 ⋯ 1 a 0 + a 1 x n + a 2 x n 2 + ⋯ + a n x n n = y n \begin{cases} 1a_0+a_1x_0+a_2x_0^2+\cdots +a_nx_0^n=y_0\\ 1a_0+a_1x_1+a_2x_1^2+\cdots +a_nx_1^n=y_1\\ \cdots\\ 1a_0+a_1x_n+a_2x_n^2+\cdots +a_nx_n^n=y_n \end{cases} 1a0+a1x0+a2x02++anx0n=y01a0+a1x1+a2x12++anx1n=y11a0+a1xn+a2xn2++anxnn=yn
  4. 用于插值规定了输入的 x i x_i xi要互不相同,因此得到的行列式D具有唯一性,且为范德蒙德行列式。
    D = ∣ 1 x 0 x 0 2 ⋯ x 0 n 1 x 1 x 1 2 ⋯ x 1 n ⋮ ⋮ ⋮ ⋱ ⋮ 1 x n x n 2 ⋯ x n n ∣ = ∏ 0 ≤ j ≤ i ≤ n ( x i − x j ) D=\begin{vmatrix} 1&x_0&x_0^2&\cdots &x_0^n\\ 1&x_1&x_1^2&\cdots &x_1^n\\ \vdots&\vdots&\vdots&\ddots&\vdots\\ 1&x_n&x_n^2&\cdots &x_n^n \end{vmatrix}=\prod_{0\le j\le i \le n}(x_i-x_j) D= 111x0x1xnx02x12xn2x0nx1nxnn =0jin(xixj)
    不难看出这个行列式唯一且不为0。
  5. 只需要解出方程组即可,比如克拉默法则:
    D i = ∣ 1 x 0 ⋯ x 0 i − 1 y 0 x 0 i + 1 ⋯ x 0 n 1 x 1 ⋯ x 1 i − 1 y 1 x 1 i + 1 ⋯ x 1 n ⋮ ⋮ ⋯ ⋮ ⋮ ⋮ ⋯ ⋮ 1 x n ⋯ x n i − 1 y n x n i + 1 ⋯ x n n ∣ D_i=\begin{vmatrix} 1&x_0&\cdots &x_0^{i-1}& y_0 &x_0^{i+1}&\cdots &x_0^n\\ 1&x_1&\cdots &x_1^{i-1}& y_1 &x_1^{i+1}&\cdots &x_1^n\\ \vdots&\vdots&\cdots&\vdots&\vdots&\vdots&\cdots&\vdots\\ 1&x_n&\cdots &x_n^{i-1}& y_n &x_n^{i+1}&\cdots &x_n^n\\ \end{vmatrix} Di= 111x0x1xnx0i1x1i1xni1y0y1ynx0i+1x1i+1xni+1x0nx1nxnn
    然后 a i = D i D a_i=\frac{D_i}{D} ai=DDi即可解出所有的系数

代码

代码里面有部分求行列式值的算法是参考此博客的,使用了里面的使用代数余子式分解的方法求解对应的值,因为本文只是讨论本方法的插值效果,故没有考虑运行速度,仅是为了简化非主要部分的代码才采用代数余子式求行列式的值的方法,如果需要提升运行速度,可以选择该博客里的另一个方法进行替代。

//多项式插值
#include<iostream>
#include<cmath>
//使用代数余子式进行求解
double determinant_value(double **D,int n)
{//递归终点if(n==1){return  D[0][0];}else if(n==2){return D[1][1]*D[0][0]-D[0][1]*D[1][0];}else{double D_value=0;for(int k=0;k<n;k++){double **D_temp=new double *[n-1];int i2=1;//由于是根据第0行第j列展开,所以原本的行列式直接从第一行开始拷贝for(int i1=0;i1<n-1;i1++){D_temp[i1]=new double[n-1];int j2=0;for(int j1=0;j1<n-1;j1++){if(j2==k){j2++;}//去除第j列D_temp[i1][j1]=D[i2][j2++];}i2++;}D_value+=(((k+2)&0x0001)?(-1):1)*D[0][k]*determinant_value(D_temp,n-1);//计算代数余子式与对应项的乘积for(int i=0;i<n-1;i++){delete[] D_temp[i];}delete[] D_temp;}return D_value;}
}
//多项式插值系数计算,输入插值坐标x,y,存储系数Di的数组和插值点数n
void polynomial_interpolation(double *x,double *y,double *D_i,int n)
{double **D=new double*[n];double D_value;double **D_temp=new double*[n];//其实这里求D可以直接利用范德蒙德行列式的性质求解for(int i = 0;i < n;i++){D[i]=new double[n];D_temp[i]=new double[n];for(int j = 0;j < n;j++){if(j==0){D[i][j]=1;}else{D[i][j]=D[i][j-1]*x[i];}}}D_value=determinant_value(D,n);//下面是为了求D_i,每次只需要修改两列数据for (int i = 0; i < n;i++){if(i==0){for (int i1 = 0; i1 < n;i1++){D_temp[i1][0]=y[i1];for(int j1 = 1;j1 < n;j1++){D_temp[i1][j1]=D[i1][j1];}}}else{for(int i2 = 0;i2 < n;i2++){D_temp[i2][i-1]=D[i2][i-1];D_temp[i2][i]=y[i2];}}//求多项式系数D_i[i] = determinant_value(D_temp, n) / D_value;}for (int i = 0; i < n; i++){delete[] D[i];delete[] D_temp[i];}delete[] D;delete[] D_temp;
}
double fx(double x)
{//这里填入被插值函数//如果插值区间包括分母为0的情况需要手动调整return sqrt(x + 3);
}
//利用得到的多项式系数计算函数值
double Dx(double x,int n,double *D_i)
{double x_temp = 1.0;double y_temp = 0;for(int j = 0;j < n;j++){y_temp += D_i[j]*x_temp;x_temp*= x;}return y_temp;
}
int main()
{int n;//插值点数std::cout<<"请输入插值点数:";std::cin>>n;double error[3]={0.0f,0.0f,0.0f};//误差评价double *x = new double [n];double *y = new double [n];double *D_i = new double [n];double a=3, b=10;//定义插值区间//生成插值数据for (int i = 0;i<n;i++){x[i] = a + (b-a)*i/(n-1);y[i] = fx(x[i]);}// std::cout<<"请输入插值点坐标x:"<<std::endl;// for(int i = 0;i < n;i++)// {//     std::cin>>x[i];// }// std::cout<<"请输入插值点坐标y:"<<std::endl;// for(int i = 0;i < n;i++)// {//     std::cin>>y[i];// }polynomial_interpolation(x, y, D_i, n);std::cout<<"插值多项式为:y(x)="<<D_i[0];for(int i = 1;i < n;i++){if(D_i[i]>0){std::cout << "+"<<D_i[i] <<"x^"<<i;}else{std::cout << D_i[i] <<"x^"<<i;}}std::cout << std::endl;//验证误差,抽取区间内的100个点进行误差检测for(int i = 0;i < 100;i++){double x_temp = a + (b-a)*i/(100-1);double y_temp = fx(x_temp);double y_temp2 = Dx(x_temp, n, D_i);if(fabs(y_temp-y_temp2)>error[0]){error[0] = fabs(y_temp-y_temp2);}error[1] += fabs((y_temp-y_temp2)/y_temp);error[2] += ((y_temp-y_temp2))*((y_temp-y_temp2));}//err[0]得到的是在区间内最大的误差//err[1]得到的是平均最大相对误差//err[2]是均方根误差error[1] = error[1]/100;error[2] = sqrt(error[2]/100);for(int i = 0;i < 3;i++){std::cout<<"误差"<<i+1<<":"<<error[i]<<std::endl;}//std::cout<<fx(3.4)<<"  "<<Dx(3.4, n, D_i)<<std::endl;delete[] x;delete[] y;delete[] D_i;return 0;
}

结果与分析

运行上面的代码如下:
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
这里的误差1为区间内最大的误差,误差2是平均相对误差,误差3是均方根误差,这里的误差评判标准仅作参考。
不难发现当插值点数为3和5时误差都非常小,可以根据精度需求进行选取,但是当插值点数为10时,误差将会非常大,所以选取合适的插值点数是十分必要的,否则会因为龙格现象,导致插值函数在插值点附近的值波动很大,从而导致误差很大。
这种插值方法是非常不建议使用的,因为随着阶数增长其计算量会非常大,在我的电脑上3和5都是瞬间计算完成,但是到了10就需要几秒了,到了12就算很久都算不出来

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/470638.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【PyQt6】QScreen 屏幕截屏

文章目录 0 环境1 简介2 QScreen 类2.1 获取 QScreen 的对象2.2 QScreen 的常见信息 3. 示例代码 0 环境 - Python 3.12.1 - PyQt6 6.6.1 pip install PyQt6 PyQt6-Qt6 6.6.1 默认安装PyQt6-sip 13.6.…

PowerShell搭建vue起始项目

Windows PowerShell搭建vue起始项目 搜索PowerShell,以管理员身份运行。 复制文件夹路径 cd 到这个文件夹位置 命令行创建项目&#xff1a;vue create 项目名 这里写自己的项目名就行&#xff0c;我写的yeb vue create yeb 创建成功后是这样的 有颜色的就是选中的&#xff…

掌握Go并发:Go语言并发编程深度解析

&#x1f3f7;️个人主页&#xff1a;鼠鼠我捏&#xff0c;要死了捏的主页 &#x1f3f7;️系列专栏&#xff1a;Golang全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&…

jacob使用教程---环境搭建及基本组成元素

参考资料: jacob的GitHub地址 jacob官网(个人感觉不重要) microsoft官方VBA文档(很重要,jacob所有的参数都来自于这里) jacob找COM组件 jacob环境配置教程 jacob将word转为各种格式 提取word中审阅内容 本人参考例子及相关资料 具有参考价值的博客 jacob操作e…

分布式文件系统 SpringBoot+FastDFS+Vue.js【一】

分布式文件系统 SpringBootFastDFSVue.js【一】 一、分布式文件系统1.1.文件系统1.2.什么是分布式文件系统1.3.分布式文件系统的出现1.3.主流的分布式文件系统1.4.分布式文件服务提供商1.4.1.阿里OSS1.4.2.七牛云存储1.4.3.百度云存储 二、fastDFS2.1.fastDSF介绍2.2.为什么要使…

【初始C++】引用的概念及使用场景、引用与指针的区别、内联函数、类型推导关键字auto、范围for循环、指针空值nullptr

目录 1.引用 1.1引用的概念 1.2引用的特性 1.3使用场景 1.4引用与指针的区别 2.内联函数 2.1内联函数的概念 2.2内联函数的特征 3.auto关键字&#xff08;C11&#xff09; 4.基于范围的for循环&#xff08;C11&#xff09; 5.指针空值nullptr&#xff08;C11&#x…

Linux释放内存

free -m是Linux上查看内存的指令&#xff0c;其中-m是以兆&#xff08;MB&#xff09;为单位&#xff0c;如果不加则以KB为单位。 如下图表示&#xff0c;&#xff08;total&#xff09;总物理内存是809MB&#xff0c;&#xff08;used&#xff09;已使用167MB&#xff0c;&…

OAuth 2.0 协议介绍【实现 GitHub 第三方登录】

OAuth&#xff08;是 Open Authorization 开放授权的缩写&#xff09;,在全世界得到广泛应用&#xff0c;目前的版本是2.0版。 本文会对OAuth 2.0的设计思路和运行流程&#xff0c;做一个简明通俗的解释&#xff0c;主要参考材料为RFC 6749。 OAuth 2.0 是一个开放标准&#…

vue3 之 倒计时函数封装

理解需求 编写一个函数useCountDown可以把秒数格式化为倒计时的显示xx分钟xx秒 1️⃣formatTime为显示的倒计时时间 2️⃣start是倒计时启动函数&#xff0c;调用时可以设置初始值并且开始倒计时 实现思路分析 安装插件 dayjs npm i dayjs倒计时逻辑函数封装 // 封装倒计时…

JVM(4)原理篇

1 栈上的数据存储 在Java中有8大基本数据类型&#xff1a; 这里的内存占用&#xff0c;指的是堆上或者数组中内存分配的空间大小&#xff0c;栈上的实现更加复杂。 以基础篇的这段代码为例&#xff1a; Java中的8大数据类型在虚拟机中的实现&#xff1a; boolean、byte、char…

Docker的常见命令以及命令别名

常见命令 命令说明docker pull拉取镜像docker push推送镜像到DockerRegistrydocker images查看本地镜像docker rmi删除本地镜像docker run创建并允许容器docker stop停止指定容器docker start启动指定容器docker restart重新启动容器docker rm删除指定容器docker ps查看容器do…

AcWing 122 糖果传递(贪心)

[题目概述] 有 n 个小朋友坐成一圈&#xff0c;每人有 a[i] 个糖果。 每人只能给左右两人传递糖果。 每人每次传递一个糖果代价为 1。 求使所有人获得均等糖果的最小代价。 输入格式 第一行输入一个正整数 n&#xff0c;表示小朋友的个数。 接下来 n 行&#xff0c;每行一个…