zoukankan      html  css  js  c++  java
  • 小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
    ", tre[find(id)].sum); //返回当前点(x,y)所在的连通块的大小
    	}
    	return 0;
    }
    
    欢迎评论交流!
  • 相关阅读:
    C# 利用StringBuilder提升字符串拼接性能
    T420 开启麦克风
    理解数据库中的undo日志、redo日志、检查点
    网络编程api总结
    源码剖析Linux epoll实现机制及Linux上惊群
    linux信号的处理--部分源码分析
    git使用笔记-提高篇-重置揭密
    git使用笔记-比较分支差异
    git使用笔记-提高篇
    ubuntu安装软件依赖解决
  • 原文地址:https://www.cnblogs.com/alking1001/p/11651039.html
Copyright © 2011-2022 走看看