题目描述
输入格式
输出格式
样例
数据范围与提示
25%算法:
要求选法,k<=10 杨辉三角打组合数表
n m很大,但w较小,稀疏图。由于k>=1所以无树的坐标直线无贡献,直接离散化,这样得到了最坏情况为w*w的图。然后脑残O(w^2)算一波前缀和,放到四个w^2数组里,再O(w^2)直接扫算贡献。。。
局限:空间复杂度太高O(5w^2)最多卡10000,时间O(w^2)
1 #include<cstdio> 2 #include<algorithm> 3 #include<cmath> 4 #include<cstring> 5 //#define MAXN 1000005 6 #define MAXK 11 7 #define MAXW 10005 8 #define ll long long 9 #define reg register 10 #define F(i,a,b) for(i=a;i<=b;++i) 11 using namespace std; 12 //对图离散化,因为k>=1,所以在某坐标直线上无树则无贡献 13 ll c[MAXW][MAXK]; 14 ll mod=2147483647; 15 int H[MAXW],L[MAXW],e[MAXW][MAXW],hs[MAXW][MAXW],ls[MAXW][MAXW],la[105][105],n,m,cx,cy; 16 int read() 17 { 18 reg char c; reg int x=0; 19 c=getchar(); 20 while(c<'0'||c>'9') c=getchar(); 21 while(c>='0'&&c<='9') x=x*10+c-48,c=getchar(); 22 return x; 23 } 24 struct POINT{ 25 int x,y; 26 }p[MAXW]; 27 void debug() 28 { 29 reg int i,j; 30 F(i,0,n) 31 { 32 F(j,0,m) 33 { 34 printf("%d ",la[i][j]); 35 } 36 puts(""); 37 } 38 puts(""); 39 F(i,1,cx) 40 { 41 F(j,1,cy) 42 { 43 printf("%d ",hs[i][j]); 44 } 45 puts(""); 46 } 47 puts(""); 48 F(i,1,cx) 49 { 50 F(j,1,cy) 51 { 52 printf("%d ",ls[i][j]); 53 } 54 puts(""); 55 } 56 } 57 int main() 58 { 59 int W,u; reg int i,j; 60 mod++; 61 n=read(); m=read(); W=read(); 62 F(i,1,W) 63 p[i].x=H[i]=read(),p[i].y=L[i]=read(); 64 sort(H+1,H+W+1); sort(L+1,L+W+1); 65 cx=unique(H+1,H+W+1)-H-1; cy=unique(L+1,L+W+1)-L-1; 66 F(i,1,W) 67 { 68 // la[p[i].x][p[i].y]=1; 69 p[i].x=lower_bound(H+1,H+cx+1,p[i].x)-H; 70 p[i].y=lower_bound(L+1,L+cy+1,p[i].y)-L; 71 e[p[i].x][p[i].y]=1; 72 } 73 u=read(); 74 c[0][0]=1; 75 F(i,1,n) 76 { 77 c[i][0]=1; 78 F(j,1,min(i,u)) c[i][j]=(c[i-1][j]+c[i-1][j-1])%mod; 79 } 80 F(i,1,cx) 81 { 82 F(j,1,cy) 83 { 84 hs[i][j]=hs[i][j-1]+e[i][j]; 85 ls[i][j]=ls[i-1][j]+e[i][j]; 86 } 87 } 88 reg int w,a,s,d; 89 reg ll ans=0; 90 F(i,1,cx) 91 { 92 F(j,1,cy) 93 { 94 if(e[i][j]) continue; 95 w=hs[i][j]; a=ls[i][j]; s=ls[cx][j]-ls[i][j]; d=hs[i][cy]-hs[i][j]; 96 if(w<u||a<u||s<u||d<u) continue; 97 // printf("%d %d %d %d %d %d ",i,j,w,a,s,d); 98 // printf(" %lld %lld %lld %lld ",c[w][u],c[a][u],c[s][u],c[d][u]); 99 ans=(ans+c[w][u]*c[a][u]%mod*c[s][u]%mod*c[d][u]%mod)%mod; 100 } 101 } 102 // printf("%lld",c[4][2]); 103 // debug(); 104 printf("%lld",ans); 105 return 0; 106 }
View Code
60%算法:
针对于上面的局限,考虑把平面问题转化为线性,优化aa[]…的预处理。
首先在离散化时顺带处理出每行每列的总树数 hs[] ls[],然后对所有树以x为第一关键字,y为第二关键字排序,这样O(w)扫描满足拓扑序直接统计即可。
然后再O(w)扫,知有贡献的墓地一定在同一行且两侧常青树>=k的段(但不一定满足列)中,每段的贡献为Σ(c[j上][k]*c[j下][k])*c[j左][k]*c[j右][k],j在同一行的两棵常青树中。
局限:时间复杂度(w^2)
100%算法:
不难发现上面的柿子可以用前缀和优化,由于带修用树状数组。
树状数组维护前i列的Σ(c[j上][k]*c[j下][k]),每扫到树更新当前列即可。由此复杂度优化到O(wlogw)。
常数黑科技优化,对2^32取模,可以用unsigned int对2^64自然溢出,最后输出再%2^32。
WA因:
1 if(ww[i]>=u&&ss[i]-1>=u) 2 { 3 add(p[i].y,-wl[p[i].y]); 4 add(p[i].y,c[ww[i]][u]*c[ss[i]-1][u]); 5 wl[p[i].y]=c[ww[i]][u]*c[ss[i]-1][u]; 6 }
若此树是该列的第一棵不能做出贡献的树,则未将其列贡献清零。
1 #include<cstdio> 2 #include<algorithm> 3 #include<cmath> 4 #include<cstring> 5 #define MAXK 11 6 #define MAXW 100005 7 #define un unsigned 8 #define ll long long 9 #define reg register 10 #define F(i,a,b) for(i=a;i<=b;++i) 11 using namespace std; 12 un int c[MAXW][MAXK],tr[MAXW]; 13 ll mod=2147483647; 14 int H[MAXW],L[MAXW],hs[MAXW],ls[MAXW],n,m,cx,cy,ww[MAXW],aa[MAXW],ss[MAXW],dd[MAXW],htot[MAXW],ltot[MAXW],W; 15 un int wl[MAXW]; 16 int read() 17 { 18 reg char c; reg int x=0; 19 c=getchar(); 20 while(c<'0'||c>'9') c=getchar(); 21 while(c>='0'&&c<='9') x=x*10+c-48,c=getchar(); 22 return x; 23 } 24 struct POINT{ 25 int x,y; 26 friend bool operator<(POINT a,POINT b) 27 { 28 return a.x==b.x?a.y<b.y:a.x<b.x; 29 } 30 }p[MAXW]; 31 void add(int x,un int w) 32 { 33 while(x<=W) 34 { 35 tr[x]+=w; 36 x+=(x&-x); 37 } 38 } 39 un int get(int x) 40 { 41 reg un int ans=0; 42 while(x>0) 43 { 44 ans+=tr[x]; 45 x-=(x&-x); 46 } 47 return ans; 48 } 49 int main() 50 { 51 int u; reg int i,j; 52 mod++; 53 n=read(); m=read(); W=read(); 54 F(i,1,W) p[i].x=H[i]=read(),p[i].y=L[i]=read(); 55 sort(H+1,H+W+1); sort(L+1,L+W+1); 56 cx=unique(H+1,H+W+1)-H-1; cy=unique(L+1,L+W+1)-L-1; 57 F(i,1,W) 58 { 59 p[i].x=lower_bound(H+1,H+cx+1,p[i].x)-H; 60 p[i].y=lower_bound(L+1,L+cy+1,p[i].y)-L; 61 ++htot[p[i].x]; 62 ++ltot[p[i].y]; 63 } 64 u=read(); 65 sort(p+1,p+W+1); 66 c[0][0]=1; 67 F(i,1,W) 68 { 69 c[i][0]=1; 70 F(j,1,min(i,u)) c[i][j]=c[i-1][j]+c[i-1][j-1]; 71 } 72 F(i,1,W) 73 { 74 ss[i]=ltot[p[i].y]-ls[p[i].y]; 75 dd[i]=htot[p[i].x]-hs[p[i].x]; 76 ww[i]=++ls[p[i].y]; 77 aa[i]=++hs[p[i].x]; 78 } 79 reg un int ans=0; 80 F(i,1,W) 81 { 82 add(p[i].y,-wl[p[i].y]); 83 add(p[i].y,c[ww[i]][u]*c[ss[i]-1][u]); 84 wl[p[i].y]=c[ww[i]][u]*c[ss[i]-1][u]; 85 if(p[i-1].x==p[i].x&&aa[i]-1>=u&&dd[i]>=u) 86 ans+=(get(p[i].y-1)-get(p[i-1].y))*c[aa[i]-1][u]*c[dd[i]][u]; 87 } 88 printf("%lld",ans%mod); 89 return 0; 90 }
AC
PS:第一次对拍 存板
1 #include<cstdio> 2 #include<algorithm> 3 #include<ctime> 4 #include<cstring> 5 //#define MAXN 1000005 6 #define MAXK 11 7 #define MAXW 100005 8 #define un unsigned 9 #define ll long long 10 #define reg register 11 #define F(i,a,b) for(i=a;i<=b;++i) 12 using namespace std; 13 14 int main() 15 { 16 reg double t1,t2; 17 while(1) 18 { 19 system("./data.out"); 20 system("./std.out"); 21 t1=clock(); 22 system("./a.out"); 23 t2=clock(); 24 if(system("diff std.ans a.ans")) 25 { 26 printf("WA %lfms ",(t2-t1)); 27 return 0; 28 } 29 else printf("AC %lfms ",(t2-t1)); 30 } 31 return 0; 32 }
对拍
1 #include<cstdio> 2 #include<cstdlib> 3 #include<algorithm> 4 #include<ctime> 5 #include<cstring> 6 #define MAXN 30000 7 #define MAXK 10 8 #define MAXW 100000 9 #define un unsigned 10 #define ll long long 11 #define reg register 12 #define F(i,a,b) for(i=a;i<=b;++i) 13 using namespace std; 14 bool e[MAXN+5][MAXN+5]; 15 int rad() 16 { 17 return rand()*rand(); 18 } 19 int main() 20 { 21 freopen("data.in","w",stdout); 22 srand(time(NULL)); 23 int n,m,w,k; 24 n=rad()%MAXN+1; m=rad()%MAXN+1; w=rad()%min(MAXW,(n+1)*(m+1))+1; k=rad()%MAXK+1; 25 printf("%d %d %d ",n,m,w); 26 reg int i,a,b; 27 F(i,1,w) 28 { 29 do{a=rad()%(n+1);b=rad()%(m+1);}while(e[a][b]); 30 e[a][b]=1; 31 printf("%d %d ",a,b); 32 } 33 printf("%d",k); 34 return 0; 35 }
数据生成