他想把1到2N這些數依次放進去。但是爲了使格子看起來優美,他想找到使每行每列都遞增的方案。只是畫了很是久,他發現方案數實在是太多了。度度熊想知道,有多少种放數字的方法能知足上面的條件?php
2 1 3
Case #1: 1 Case #2: 5Hint對於第二組例子。共5種方案。詳細方案爲:
經過打表得出前7項分別爲1,2,5,14,42,132,429,可知答案爲卡特蘭數h(n)=C(2n,n)/(n+1)=h(n-1)*(4*n-2)/(n+1)。ios
一開始採用組合數分解素因子+二分求冪求組合數取模,可是會TLE,組合數求模相關知識http://hi.baidu.com/aekdycoin/item/e051d6616ce60294c5d249d7。渣代碼例如如下:post
#include <stdio.h> #include <string> #include <iostream> #include<vector> #include<cmath> #include<algorithm> using namespace std; const int N = 2000005; const int n=148955; bool a[N];//a[]的長度比pr[]的長度長得多 int pr[n]; #define MOD 1000000007 int num; void Prime2() { memset(a, 0, N*sizeof(a[0])); int i, j; num = 0; a[0]=a[1]=1; for(i = 2; i < N; ++i) { if(!(a[i])) pr[num++] = i; for(j = 0; (j<num && i*pr[j]<N); ++j) { a[i*pr[j]] = 1; if(!(i%pr[j])) break; } } } int val[n],len; void calcJC(int n,int id,int flag){ int ans=0,y,p=pr[id]; while(n){ y=n/p; ans+=y; n=y; } val[id]=val[id]+ans*flag; } __int64 extgcd(__int64 a,__int64 b,__int64 &x,__int64 &y) { if(b==0) { x=1,y=0; return a; } __int64 r=extgcd(b,a%b,x,y); __int64 t=x;x=y;y=t-a/b*y; return r; } int MPow(int p,int e){ if(e==0)return 1; else if(e==1)return p; int t=p,ans=1; while(e){ if(e&1)ans=(ans*t)%MOD; t=(t*t)%MOD; e>>=1; } return ans; } int main() { Prime2(); int txt,l=1,k,i; __int64 ans,x,y; scanf("%d",&txt); while(txt--){ scanf("%d",&k); memset(val,0,sizeof(val)); for(i=0;pr[i]<=2*k;++i) calcJC(2*k,i,1); for(i=0;pr[i]<=k;++i) calcJC(k,i,-2); ans=1; for(i=0;pr[i]<=2*k;++i){ // if(val[i]>0)printf("%d^%d ",pr[i],val[i]); ans=(ans*MPow(pr[i],val[i]))%MOD; } extgcd(k+1,MOD,x,y); x=(x+MOD)%MOD; ans=(ans*x)%MOD; printf("%I64d\n",ans); } return 0; }
#include <stdio.h> #include <string> #include <iostream> #include<vector> #include<cmath> #include<algorithm> using namespace std; #define MOD 1000000007 const int N = 1000001; int a[N]; __int64 extgcd(__int64 a,__int64 b,__int64 &x,__int64 &y) { if(b==0) { x=1,y=0; return a; } __int64 r=extgcd(b,a%b,x,y); __int64 t=x;x=y;y=t-a/b*y; return r; } void calcCATALAN(int n){ __int64 x,y; a[1]=1; int i; for(i=2;i<n;++i){ x=a[i-1]; a[i]=(x*(4*i-2))%MOD; extgcd(i+1,MOD,x,y); x=(x+MOD)%MOD; a[i]=(a[i]*x)%MOD; } } int main() { calcCATALAN(N); int txt,l=1,k; scanf("%d",&txt); while(txt--){ scanf("%d",&k); printf("Case #%d:\n",l++); printf("%d\n",a[k]); } return 0; }