小z的洞穴之旅 QDUOJ 并查集+连通块
小z的洞穴之旅 QDUOJ 并查集+连通块
题意
小 z 同学在某个闲暇的周末决定去野外探险一波,结果在丛林深处中误打误撞进入了一个神秘的洞穴,虽然洞穴中光线昏暗,但小 z 凭借其敏锐的眼力立刻辨认出这是一个迷宫状洞穴,并且他还发现了一个现象:该洞穴中时不时会有一个墙块自行坍塌,每个墙体坍塌后其所在单元格即变为空地,其坍塌过程中所产生的尘土也会随之传到该墙体相连的各个空地处,于是他很好奇,对于每一次墙块的坍塌,所产生的尘土会遍及到多大的空白区域?
解题思路
这个题很像《啊哈!算法》中的岛屿问题,但是如果每次有墙倒塌后都进行BFS一次的话会超时,所以这里要先进行预处理,这里预处理是用bfs来扫,把联通的块使用并查集归并到一起(对于二维的数据,需要转化为一维的数据,代码中的getid函数就是干这个的)。之后每次有墙倒塌就查看周围四个位置, 详情见代码实现。
在此也感谢ZYB学长的题解!当时没有尝试做,很忏愧。
代码实现
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
using namespace std;
const int maxn=5e3+7;
struct note{
int fa, sum;
}tre[maxn*maxn];
struct node{
int x, y;
};
int n, m;
char mp[maxn][maxn];
int go[4][2]={1, 0, 0, 1, -1, 0, 0, -1};
queue<node> q;
inline int getid(int x, int y)// getid()将二维点(x,y)映射为一维的数值
{
return (x-1)*m + y;
}
int find(int x) //寻找父节点,并且进行路径压缩
{
if(tre[x].fa==x) return x;
else return tre[x].fa = find(tre[x].fa);
}
void merge(int u, int v) //这里合并时按照各自集合中点的个数来进行合并
{
u=find(u);
v=find(v);
if(u==v) return ;
if(tre[u].sum >= tre[v].sum)
{
tre[v].fa=u;
tre[u].sum+=tre[v].sum;
}
else {
tre[u].fa=v;
tre[v].sum+=tre[u].sum;
}
}
void bfs(int x, int y, int rt) //预处理中的一部分
{
while(!q.empty()) q.pop() ;
node h={x, y}, tmp;
int tx, ty, id;
q.push(h);
while(!q.empty())
{
h=q.front();
q.pop() ;
for(int i=0; i<4; i++)
{
tx=h.x+go[i][0];
ty=h.y+go[i][1];
if(tx<=0 || tx>n || ty<=0 || ty>m || mp[tx][ty]=='#')
continue;
id=getid(tx, ty);
if(tx==x && ty==y || tre[id].fa!=id) continue;
merge(rt, id);
tmp.x=tx;
tmp.y=ty;
q.push(tmp);
}
}
}
void init() //预处理的主程序
{
int id;
for(int i=1; i<=n; i++)
for(int j=1; j<=m; j++)
{
id=getid(i, j);
tre[id].fa=id; //默认父节点都是自己
tre[id].sum=1;//默认个数都是1
}
for(int i=1; i<=n; i++)
for(int j=1; j<=m; j++)
{
id=getid(i, j);
// fa[tmp]!=tmp即表示该块已被遍历过
if(mp[i][j]=='#' || tre[id].fa != id) continue;
bfs(i, j, id);
}
}
int main()
{
int q, x, y, tx, ty, id, tmp;
scanf("%d%d", &n, &m);
for(int i=1; i<=n; i++)
scanf("%s", mp[i]+1);
init();
scanf("%d", &q);
while(q--)
{
scanf("%d%d", &x, &y);
id=getid(x, y);
//尝试将(x,y)点与周围上下左右四个区域合并
for(int i=0; i<4; i++)
{
tx=x+go[i][0];
ty=y+go[i][1];
if(tx<=0 || tx>n || ty<=0 || ty>m || mp[tx][ty]=='#')
continue;
tmp=getid(tx, ty);
merge(id, tmp);
}
mp[x][y]='.'; //最后还需要维护原始的mp[][]
printf("%d\n", tre[find(id)].sum); //返回当前点(x,y)所在的连通块的大小
}
return 0;
}
小z的洞穴之旅 QDUOJ 并查集+连通块的更多相关文章
- 【BZOJ】2049: [Sdoi2008]Cave 洞穴勘测(lct/并查集)
http://www.lydsy.com/JudgeOnline/problem.php?id=2049 bzoj挂了..在wikioi提交,,1A-写lct的速度越来越快了-都不用debug-- 新 ...
- BZOJ_2049_[Sdoi_2008]_Cave_洞穴勘测_(LCT/并查集)
描述 http://www.lydsy.com/JudgeOnline/problem.php?id=2049 给出一个森林,起始互不相连,现在有link和cut两种操作,问x,y是否在一棵树里. 分 ...
- 小Z的城市之旅
题目大意: 从(0,0)走到(x,y),每次可以向上下左右走一格,问走r步刚好到(x,y)的方案数. 思路: 找规律发现答案就是C((r+x-y)/2,r)*C((r-x-y)/2,r). #incl ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) [莫队算法]【学习笔记】
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 7687 Solved: 3516[Subm ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose)
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 7676 Solved: 3509[Subm ...
- BZOJ2038: [2009国家集训队]小Z的袜子(hose)
Time Limit: 20 Sec Memory Limit: 259 MB Description 作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿.终于有一天, ...
- Bzoj 2038---[2009国家集训队]小Z的袜子(hose) 莫队算法
题目链接 http://www.lydsy.com/JudgeOnline/problem.php?id=2038 Description 作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色 ...
- 莫队算法 2038: [2009国家集训队]小Z的袜子(hose)
链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2038 2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 ...
- BZOJ 2038 小z的袜子 & 莫队算法(不就是个暴力么..)
题意: 给一段序列,询问一个区间,求出区间中.....woc! 贴原题! 作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿.终于有一天,小Z再也无法忍受这恼人的找袜子过 ...
随机推荐
- SQL复杂筛选
SELECT A.MATERIALID,A.MATERIALNAME,ISNULL(A.COMPIDSEQ,'') COMPIDSEQ,ISNULL(A.SUPPLYID,'') SUPPLYID,S ...
- 对Serverless的研究
1. 引言 Serverless 是一种 “无服务器架构”,让用户无需关心程序运行环境.资源及数量,只要将精力 Focus 到业务逻辑上的技术. 现在公司已经实现 DevOps 化,正在向 Serve ...
- zabbix 磁盘自动发现脚本
##需要在zabbix界面配置宏变量===>正则来匹配磁盘 disk_discovery.sh ———————————————————————————————————————————————— ...
- ssh复制公钥成功后仍需输入密码
1,网上说权限问题 登录流程: 被登录机器的文件权限: //用户权限 chmod 700 /home/username //.ssh文件夹权限 chmod 700 ~/.ssh/ // ~/.ssh/ ...
- SQLite为何要用C语言来开发?
SQLite 选择 C 语言的理由是?为什么不选择 Go 或者 Rust? C 语言是最好的 SQLite 在 2000 年 5 月 29 日发布,并一直使用 C 语言实现.C 语言一直是实现 SQL ...
- Codeforces 963A Alternating Sum ( 思维 && 数论 )
题意 : 题目链接 分析 : Tutorial 讲的很清楚 至于为什么这样去考虑 算是一个经验问题吧 如果一个问题要你给出模意义下的答案 就多考虑一下答案是要用逆元构造出来 也就说明有除法的存在 那么 ...
- sublime text3安装格式化代码插件
1.代码提示插件:sublimeCodeIntel a)在Sublime Text 3中,按下Ctrl+Shift+P调出命令面板;b)输入install 调出 Install Package 选项并 ...
- 基本的axios用法
首先安装axios: 1):npm install 2):npm install vue-axios --save 3):npm install qs.js --save //它的作用是能把json格 ...
- [CSP-S模拟测试]:军训队列(DP+乱搞)
题目描述 有$n$名学生参加军训,军训的一大重要内容就是走队列,而一个队列的不规整程度是该队中最高的学生的身高与最矮的学生的身高差值的平方.现在要将$n$名参加军训的学生重新分成$k$个队列,每个队列 ...
- mysql 安装教程(详细说明)
如果你装过,一定要先卸载干净,并且重启重新装.卸载教程(保证成功)https://www.cnblogs.com/qzhc/p/11354678.html 大家都知道MySQL是一款中.小型关系型数据 ...