洛谷-P9165 「INOH」Round 1 - 意外 题解
xiaoniu142857 · · 题解
Solution
由题意得,传输的数组长度必须
最朴素的容错方式是增加冗余。假设需要传递
最直接的想法是直接传原数组,每个数传
能否找到一种方法,使得只需要任意
有两种构造多项式的方法:
- 数组作为点值:解码器需要做
N 次O(N^2) 插值,总时间复杂度O(N^3) 。 - 数组作为系数:拉格朗日插值提供了
O(N^2) 的点值转系数算法。
综上,我们采用方法 2。
取
Code
#include <bits/stdc++.h>
#define rep(i,a,b) for(int i(a);i<b;++i)
#define per(i,a,b) for(int i(a);i>b;--i)
#define rept(i,a,b) for(int i(a);i<=b;++i)
#define pert(i,a,b) for(int i(a);i>=b;--i)
#define ll long long
#define eb emplace_back
using namespace std;
const int N=100,S=150,K=5;
const ll P=998244353;
ll ksm(ll x,ll y){
ll res=1;
while(y){
if(y&1) (res*=x)%=P;
(x*=x)%=P,y>>=1;
}
return res;
}
vector<int> Encode(vector<int> vec){
vector<int> res;
res.reserve(S*K);
rept(x,1,S){
ll y=0;
pert(i,N-1,0) y=(y*x+vec[i])%P;
rep(i,0,K) res.eb(y);
}
return res;
}
vector<ll> lagrange(const vector<ll> &x,const vector<ll> &y){
int n=x.size();
vector<ll> p(n+1,0),q(n+1,0),res(n,0);
p[0]=1;
rep(i,0,n){
pert(j,i,0){
(p[j+1]+=p[j])%=P;
(p[j]*=-x[i])%=P;
}
}
rep(i,0,n){
ll a=1,rem=p[n];
rep(j,0,n) if(i^j) (a*=x[i]-x[j])%=P;
a=ksm(a,P-2)*y[i]%P;
pert(j,n-1,0){
q[j]=rem;
(rem=p[j]+rem*x[i]%P)%=P;
}
rep(j,0,n) (res[j]+=a*q[j]%P)%=P;
}
rep(i,0,n) (res[i]+=P)%=P;
return res;
}
vector<int> Decode(vector<int> vec){
vector<ll> x,y;
x.reserve(N),y.reserve(N);
rept(i,1,S){
map<int,int> mp;
rep(j,K*(i-1),K*i){
++mp[vec[j]];
if(mp[vec[j]]>=2){
x.eb(i),y.eb(vec[j]);
break;
}
}
if(x.size()>=N) break;
}
vector<ll> res=lagrange(x,y);
return vector<int>(res.begin(),res.end());
}