首页 > 代码库 > BZOJ 1057 棋盘制作(最大黑白相间子矩阵)
BZOJ 1057 棋盘制作(最大黑白相间子矩阵)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1057
题意:给出一个只包含01的矩阵。找出一个01相间的最大的正方形子矩阵?找出一个01相间的最大的长方形子矩阵?
思路:一行一行扫描,对于某一行,记录每列 向上延伸的最大长度,记为h。然后再求出每列向右向左延伸的最大距离。比如2122345421,第4个位置的2向左延伸到3向右延伸到9。我们就是要求 出这个东西。我们用类似并查集思想。以向左为例。L[j]初始时指向j。从左向右扫描,对于j,令k=j-1,若(i,k)和(i,k+1)格子颜色不用 且h[k]>=h[j],那么令L[j]=L[k],接着令k=L[j]-1继续判断。我们求出这个L和R数组后只要再扫一遍,h[j]就是 高,R[j]-L[j]+1就是宽,更新答案即可。
int n,m,g[N][N],h[N];int Max,MAX;void deal(){ clr(h,0); int i,j,k,left,right,L[N],R[N],x,y; FOR0(i,n) { FOR0(j,m) { if(i==0||g[i][j]==g[i-1][j]) h[j]=1; else h[j]++; } FOR0(j,m) L[j]=R[j]=j; FOR1(j,m-1) { k=j-1; while(k>=0&&h[k]>=h[j]&&g[i][k]!=g[i][k+1]) { L[j]=L[k]; k=L[j]-1; } } FORL0(j,m-2) { k=j+1; while(k<=m-1&&h[k]>=h[j]&&g[i][k]!=g[i][k-1]) { R[j]=R[k]; k=R[j]+1; } } FOR0(j,m) { x=h[j]; y=R[j]-L[j]+1; if(x>y) swap(x,y); if(x==y) { if(x*y>MAX) MAX=x*y; if(y*x-x>Max) Max=y*x-x; } else { if(x*y>Max) Max=x*y; else if(x*x>MAX) MAX=x*x; } } }}int main(){ RD(n,m); int i,j; FOR0(i,n) FOR0(j,m) RD(g[i][j]); deal(); PR(MAX); PR(Max); return 0;}
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。