题意:一段长为 i 的项链有 a[i] 种装饰方式,问长度为n的相连共有多少种装饰方式
分析:采用dp做法,dp[i]=∑dp[j]*a[i-j]+a[i],(1<=j<=i-1)
然后对于这种递推式,也就是dp[i]等于前j个dp数组和a数组的卷积,然后可看所有的
一看n是1e5,所以暴力超时,然后采用cdq分治加速,这种卷积递推通常采用cdq分治加速
cdq的话很简单了,就是先递归左边,算左对右的贡献,递归右边就行,一半一半更新
#include <cstdio>
#include <vector>
#include <stack>
#include <cmath>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 1e5+;
const int mod = ;
typedef long long LL;
const double pi = acos(-1.0);
int a[N],dp[N],n;
struct complex{
double r,i;
complex(double R=,double I=){
r=R;i=I;
}
complex operator+(const complex &a)const{
return complex(r+a.r,i+a.i);
}
complex operator-(const complex &a)const{
return complex(r-a.r,i-a.i);
}
complex operator*(const complex &a)const{
return complex(r*a.r-i*a.i,r*a.i+i*a.r);
}
}x[N*],y[N*];
void change(complex x[],int len){
int i,j,k;
for(i=,j=len/;i<len-;++i){
if(i<j)swap(x[i],x[j]);
k=len/;
while(j>=k){j-=k;k>>=;}
if(j<k)j+=k;
}
}
void fft(complex x[],int len,int on){
change(x,len);
for(int i=;i<=len;i<<=){
complex wn(cos(-on**pi/i),sin(-on**pi/i));
for(int j=;j<len;j+=i){
complex w(,);
for(int k=j;k<j+i/;++k){
complex u = x[k];
complex t = w*x[k+i/];
x[k]=u+t;
x[k+i/]=u-t;
w=w*wn;
}
}
}
if(on==-)for(int i=;i<len;++i)x[i].r/=len;
}
void up(int &x){
x%=mod;
}
void cdq(int l,int r){
if(l==r){dp[l]+=a[l];up(dp[l]);return;}
int mid=l+r>>;
cdq(l,mid);
int len=;
while(len<=(r-l+))len<<=;
for(int i=;i<len;++i)x[i]=y[i]=complex(,);
for(int i=l;i<=mid;++i)x[i-l]=complex(dp[i],);
for(int i=;i<=r-l+;++i)y[i-]=complex(a[i],);
fft(x,len,);fft(y,len,);
for(int i=;i<len;++i)x[i]=x[i]*y[i];
fft(x,len,-);
for(int i=mid+;i<=r;++i)
dp[i]+=(int)(x[i-l-].r+0.5),up(dp[i]);
cdq(mid+,r);
}
int main(){
while(~scanf("%d",&n),n){
for(int i=;i<=n;++i){
scanf("%d",&a[i]);up(a[i]);dp[i]=;
}
cdq(,n);
printf("%d\n",dp[n]);
}
return ;
}