其他分享
首页 > 其他分享> > Codeforces 1383D - Rearrange(构造)

Codeforces 1383D - Rearrange(构造)

作者:互联网

Codeforces 题面传送门 & 洛谷题面传送门

一道不算困难的构造,花了一节英语课把它搞出来了,题解简单写写吧(

考虑从大往小加数,显然第三个条件可以被翻译为,每次加入一个元素,如果它所在的行/列存在元素,那么它必须为这一行/列所在的元素相邻,因此我们考虑这样构造,当我们加入一个数 \(v\) 时,分以下几种情况考虑:

时间复杂度 \(\mathcal O(nm)\)

const int MAXN=250;
const int dx[]={1,0,-1,0};
const int dy[]={0,1,0,-1};
int n,m,a[MAXN+5][MAXN+5],vis1[MAXN*MAXN+5],vis2[MAXN*MAXN+5];
int X=0,Y=0,b[MAXN+5][MAXN+5];queue<pii> q;
bool check(int x,int y){
	if(x<1||x>n||y<1||y>m) return 0;
	if(b[x][y]) return 0;
	int cnt=0;
	for(int i=0;i<4;i++) cnt+=(b[x+dx[i]][y+dy[i]]>0);
	return cnt==2;
}
void relax(int x,int y){
	for(int i=0;i<4;i++) if(check(x+dx[i],y+dy[i])) q.push(mp(x+dx[i],y+dy[i]));
}
int main(){
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) scanf("%d",&a[i][j]);
	for(int i=1;i<=n;i++){
		int mx=0;
		for(int j=1;j<=m;j++) chkmax(mx,a[i][j]);
		vis1[mx]=1;
	}
	for(int i=1;i<=m;i++){
		int mx=0;
		for(int j=1;j<=n;j++) chkmax(mx,a[j][i]);
		vis2[mx]=1;
	}
	for(int i=n*m;i;i--){
		if(vis1[i]&&vis2[i]) b[++X][++Y]=i,relax(X,Y);
		else if(vis1[i]) b[++X][Y]=i,relax(X,Y);
		else if(vis2[i]) b[X][++Y]=i,relax(X,Y);
		else{
			while(1){
				pii p=q.front();q.pop();
				if(!b[p.fi][p.se]){
					b[p.fi][p.se]=i;
					relax(p.fi,p.se);
					break;
				}
			}
		}
	}
	for(int i=1;i<=n;i++) for(int j=1;j<=m;j++)
		printf("%d%c",b[i][j]," \n"[j==m]);
	return 0;
}

标签:Rearrange,int,最大值,矩阵,Codeforces,1383D,相邻,MAXN,我们
来源: https://www.cnblogs.com/ET2006/p/Codeforces-1383D.html