Codeforces 1383D - Rearrange(构造)
一道不算困难的构造,花了一节英语课把它搞出来了,题解简单写写吧(
考虑从大往小加数,显然第三个条件可以被翻译为,每次加入一个元素,如果它所在的行/列存在元素,那么它必须为这一行/列所在的元素相邻,因此我们考虑这样构造,当我们加入一个数 \(v\) 时,分以下几种情况考虑:
- 如果 \(v\) 在原矩阵中既是行的最大值,也是列的最大值,那我们新开一行一列并将这个元素塞进去。即我们动态维护一个 \(R,C\) 表示目前有 \(R\) 行 \(C\) 列有元素,那么遇到这样的 \(v\),我们令 \(R,C\) 都加一然后令 \(b_{R,C}=v\) 即可。
- 如果 \(v\) 在原矩阵中只是行的最大值不是列的最大值,那么我们只令 \(R\) 加 \(1\) 而不用令 \(C\) 加 \(1\),然后还是令 \(b_{R,C}=v\)。
- 如果 \(v\) 在原矩阵中只是列的最大值不是行的最大值,与上面的情况不同之处在于,这次我们要令 \(C\) 加 \(1\) instead of \(R\)。
- 如果 \(v\) 在原矩阵中既不是行的最大值也不是列的最大值,那么我们就找到一个位置 \((x,y)\) 满足 \((x,y)\) 上未填上数,并且 \((x,y)\) 恰好与两个已经填上格子的位置相邻,并令 \(b_{x,y}=v\)。可以证明我们总能找到这样的位置,因为在值最大的 \(RC+1\) 个格子中必然要么有超过 \(R\) 个行最大值,要么有超过 \(C\) 个列最大值,因此不会出现没地方填的情况。而显然如果前 \(R\) 行 \(C\) 列没有填满,由于我们构造的特殊性,我们总能找到一个空格子满足其与 \(2\) 个填上值的位置相邻,因此我们的构造总是合法的。这一部分可以通过维护一个队列,每次新填上一个值就遍历一遍与其相邻的格子,检验其是否与 \(2\) 个填上值的位置相邻来实现。
时间复杂度 \(\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;
}
Codeforces 1383D - Rearrange(构造)的更多相关文章
- codeforces 1041 e 构造
Codeforces 1041 E 构造题. 给出一种操作,对于一棵树,去掉它的一条边.那么这颗树被分成两个部分,两个部分的分别的最大值就是这次操作的答案. 现在给出一棵树所有操作的结果,问能不能构造 ...
- Codeforces - 474D - Flowers - 构造 - 简单dp
https://codeforces.com/problemset/problem/474/D 这道题挺好的,思路是这样. 我们要找一个01串,其中0的段要被划分为若干个连续k的0. 我们设想一个长度 ...
- Codeforces Global Round 8 B. Codeforces Subsequences(构造)
题目链接:https://codeforces.com/contest/1368/problem/B 题意 构造最短的至少含有 $k$ 个 $codeforces$ 子序列的字符串. 题解 如下表: ...
- Codeforces 410C.Team[构造]
C. Team time limit per test 1 second memory limit per test 256 megabytes input standard input output ...
- Codeforces 716C[数论][构造]
/* CF傻逼构造题 某人要经过n回合游戏,初始分值是2,等级为1. 每次有两种操作 1.无条件,分值加上自己的等级数. 2.当目前的数字是完全平方数并且该数字开方以后是等级数加1的整数倍,那么可以将 ...
- Tea Party CodeForces - 808C (构造+贪心)
Polycarp invited all his friends to the tea party to celebrate the holiday. He has ncups, one for ea ...
- Codeforces.578E.Walking(构造)
题目链接 \(Description\) 给定一个长为\(n\)的足迹序列(只包含\(L,R\)两种字符),你需要\(LRLRLR...\)这样交替在\(L\)和\(R\)上走(第一步可以选择\(L\ ...
- New Roads CodeForces - 746G (树,构造)
大意:构造n结点树, 高度$i$的结点有$a_i$个, 且叶子有k个. 先确定主链, 然后贪心放其余节点. #include <iostream> #include <algorit ...
- Subordinates CodeForces - 737C (树,构造)
大意: 求构造一棵树, 每个节点回答它的祖先个数, 求最少打错次数. 挺简单的一个构造, 祖先个数等价于节点深度, 所以只需要确定一个最大深度然后贪心即可. 需要特判一下根的深度, 再特判一下只有一个 ...
随机推荐
- 以太坊web3开发初步学习
以太坊web3开发初步学习 此文是对https://learnblockchain.cn/2018/04/15/web3-html/的学习再理解. 以太坊智能合约通过使用web3.js前端和智能合约交 ...
- Go语言核心36讲(Go语言进阶技术四)--学习笔记
10 | 通道的基本操作 作为 Go 语言最有特色的数据类型,通道(channel)完全可以与 goroutine(也可称为 go 程)并驾齐驱,共同代表 Go 语言独有的并发编程模式和编程哲学. D ...
- 痞子衡嵌入式:超级下载算法RT-UFL v1.0在IAR EW for Arm下的使用
痞子衡主导的"学术"项目 <RT-UFL - 一个适用全平台i.MXRT的超级下载算法设计> v1.0 版发布近 4 个月了,部分客户已经在实际项目开发调试中用上了这个 ...
- 通过简单例子 | 快速理清 UML 中类与类的六大关系
关于封面:我想我们都会离开 类与类之间的六大关系 泛化 ( Generalization ) ---> 表继承关系 实现 ( Realization ) 关联 ( Association ) 聚 ...
- Java:并发笔记-09
Java:并发笔记-09 说明:这是看了 bilibili 上 黑马程序员 的课程 java并发编程 后做的笔记 7. 共享模型之工具-2 原理:AQS 原理 对于 AQS 的原理这部分内容,没很好的 ...
- 极简实用的Asp.NetCore框架再新增商城模块
概述 关于这个框架的背景,在前面我已经交代过了.不清楚的可以查看这个链接 1.极简实用的Asp.NetCore模块化框架决定免费开源了 2.极简实用的Asp.NetCore模块化框架新增CMS模块 算 ...
- 2021.9.26考试总结[NOIP模拟62]
T1 set 从\(0\)到\(n\)前缀余数有\(n+1\)个,但只有\(n\)种取值,找到一样的两个输出区间即可. \(code:\) T1 #include<bits/stdc++.h&g ...
- 2021.10.7 NKOJ周赛总结
Ⅰ. 自描述序列 问题描述: 序列 1,2,2,1,1,2,1,2,2,1,2,2,1,1,2,1,1,2,2,1,... 看似毫无规律,但若我们将相邻的数字合并 : 1,22,11,2,1,22,1 ...
- Spring Security:Servlet 过滤器(三)
3)Servlet 过滤器 Spring Security 过滤器链是一个非常复杂且灵活的引擎.Spring Security 的 Servlet 支持基于 Servlet 过滤器,因此通常首先了解过 ...
- linux tr
转载:tr命令_Linux tr 命令用法详解:将字符进行替换压缩和删除 (linuxde.net) tr命令 文件过滤分割与合并 tr命令可以对来自标准输入的字符进行替换.压缩和删除.它可以将一组字 ...