多项式插值
一个 次多项式,可以用 个点 表示。
知道了某个多项式上 个点的点值,可以用拉格朗日插值公式还原出多项式,或者求给定 的函数值。朴素做法时间复杂度 。可以运用分治+NTT优化到 。如果给定要求的点是一个点,并且你可以选择一段有特殊性质的点值(例如 ),可以用预处理的方法做到线性快速插值。
基本柿子
根据“第 项点的值为 ,其他为 ”构造。注意下方为 ,推的时候不要犯迷糊。
显然是一个 次多项式。
注意负数。
P4781
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define f(i, a, b) for(int i = (a); i <= (b); i++)
#define cl(i, n) i.clear(),i.resize(n);
#define endl '\n'
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int, int> pii;
const int inf = 1e9;
//#define cerr if(false)cerr
#define watch(x) cerr << (#x) << ' '<<'i'<<'s'<<' ' << x << endl
void cmax(int &x, int y) {if(x < y) x = y;}
void cmin(int &x, int y) {if(x > y) x = y;}
const int mod=998244353;
int qpow(int n,int k){
int ans=1;
while(k){
if(k&1)ans=ans*n%mod;
n=n*n%mod;
k>>=1;
}
return ans;
}
int x[100010],y[100010];
//调不出来给我对拍!
signed main() {
ios::sync_with_stdio(0);
cin.tie(NULL);
cout.tie(NULL);
//time_t start = clock();
//think twice,code once.
//think once,debug forever.
int n,k;cin>>n>>k;
f(i,1,n)cin>>x[i]>>y[i];
int ans=0;
f(i,1,n){
int tmp=1;
f(j,1,n){
if(j==i)continue;
tmp=tmp*(k-x[j])%mod+mod;tmp%=mod;
tmp=tmp*qpow(x[i]-x[j],mod-2)%mod+mod;tmp%=mod;
}
tmp=tmp*y[i]%mod;
ans=ans+tmp;
ans%=mod;
}
cout<<ans<<endl;
//time_t finish = clock();
//cout << "time used:" << (finish-start) * 1.0 / CLOCKS_PER_SEC <<"s"<< endl;
return 0;
}
快速求点值
如果是单点求值,并且自由选点,那么选择 ,原式可以预处理前缀后缀相关柿子做到每一项 计算。
例题
【题意】
求
其中
【分析】
首先有个观察得到的性质, 次方和是 次多项式。
然后前 项可以递推得到。
于是插值可以做。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 记一次.NET内存居高不下排查解决与启示