菁菁堂有一块数字格,那是王解体最喜欢去的地方。
传说中,这条气势磅礴的数字格,有N行N列,每一个格子里均有一个数。
敢于挑战自己的王解体决定来挑战这道通过率为百分之九十九的题目。
格子的第一行及第一列均是给定的:
F[k,1]=lk
F[1,k]=tk
对于其他格子,满足递推式:
F[i,j]=a*F[i,j-1]+b*F[i-1,j]+c
不出所料,当王解体能得到 F[n,n](mod 1000003)时,通过率将达到百分之百。
这道题可以用FFT打,但是我不会(……),所以用了另外一种巧妙的方法。
首先通过比较暴力的公式推导可以发现,特殊项的贡献是可以直接算出来的,而常数项c的解决会有一些麻烦,我们先把不含c的算出来,在推导c的贡献时,可以从系数入手,最左上角的c的系数肯定是1,向右拓展时是a的幂,向下拓展时是b的幂。由于每一个格子的数受它的上面和左边的数影响,若是斜向来看,每一斜列的系数和是上一列的(a+b)倍。(详见图,a=3,b=5时)
黄色块的和为: 1∗(3+5) ,灰色块的和为: 1∗(3+5)2 ,蓝色块的和为: 1∗(3+5)3 。
但是当超过对角线时,就会有两块多余的被计入系数和中(图中的绿块为多余块)。
怎么减去呢?我们可以发现实际上就等于灰色块对下一斜列的贡献,灰色块的计算是很容易的,这里就不推导了,用组合数再乘上a的幂和b的幂就可以了。最后c的贡献就是矩阵的系数和乘上c。
#include
#include
#include
#include
#include
using namespace std;
#define fo(i,a,b) for(i=a;i<=b;i++)
typedef long long ll;
const int N=2e5+5,mo=1e6+3;
int n,a,b,c,i,j;
ll x[N],y[N],ans,Cn,Cm,C,jc[N*2],ni[N*2],ca,cb,t,tot,num;
ll ksm(int x,int y){
if(y==0) return 1;
if(y%2) return (ksm(x,y-1)*x)%mo;
t=ksm(x,y/2);t=t*t%mo;
return t;
}
ll sgm(ll x,ll y){
return jc[y]*ni[x]%mo*ni[y-x]%mo;
}
int main(){
freopen("matrix.in","r",stdin);
freopen("matrix.out","w",stdout);
jc[0]=ni[0]=1;
fo(i,1,2*N) jc[i]=(jc[i-1]*i)%mo,ni[i]=ksm(jc[i],mo-2);
while(scanf("%d%d%d%d",&n,&a,&b,&c)!=EOF){
ans=0;
fo(i,1,n) scanf("%lld",&y[i]);
fo(i,1,n) scanf("%lld",&x[i]);
Cn=0;Cm=n-2;ca=ksm(a,n-1);cb=ksm(b,n-1);
fo(i,1,n-1){
C=sgm(Cn,Cm);
Cn++,Cm++;
ans=(ans+(C*ca%mo)*y[n-i+1])%mo;
ans=(ans+(C*cb%mo)*x[n-i+1])%mo;
ca=ca*b%mo,cb=cb*a%mo;
}
n--;
tot=0;num=1;
fo(i,1,n) tot=(tot+num)%mo,num=num*(a+b)%mo;
ca=ksm(a,n),num-=ca;
cb=ksm(b,n),num-=cb;
tot=(tot+num)%mo,num=num*(a+b)%mo;
fo(i,1,n-1){
C=sgm(i,n+i-1);
ca=C*ksm(a,n-1)%mo*ksm(b,i)%mo*a%mo;
num-=ca;if(num<0) num+=mo;
cb=C*ksm(a,i)%mo*ksm(b,n-1)%mo*b%mo;
num-=cb;if(num<0) num+=mo;
tot=(tot+num)%mo,num=num*(a+b)%mo;
}
ans=(ans+tot*c)%mo;
while(ans<0) ans+=mo;
printf("%lld\n",ans);
}
}