题目链接

大致题意

相邻格子不同为连通,计算每个点所在的连通块大小。

想法

我采用了并查集的做法。

开一个辅助数组记录连通块大小,每次合并的时候更新父亲节点的大小即可。

一个点先与它上面的点判定,若判定连通则加入上方点所属的块中。

再与左边的点判定,若连通则再将两个块合并。

总体复杂度O(n3)O(n^3)O(n3),其中并查集不加优化复杂度为O(n)O(n)O(n),使用路径压缩+按节点大小合并优化并查集,将并查集复杂度降低为近似O(1)O(1)O(1),总体复杂度近似为O(n2)O(n^2)O(n2)。

并查集优化

路径压缩

每次查找的时候,将路径上的所有儿子的父亲改写为最原始的祖宗。这样下次就可以直接找到祖宗。

实现:

inline int find(const int &x)
{
return fa[x] == x ? x : fa[x] = find(fa[x]);//路径压缩
}

通俗写法:

inline int find(const int &x)
{
if(fa[x] == x)//找到无父亲的节点,即为祖宗
return x;
fa[x] = find(fa[x]);
return fa[x];
}

按秩合并

思考:

有2个块需要合并,那么将小块接在大块的后面能够让整体复杂度更优。

可以将每个祖宗及其所有儿子看做一棵树,那么节点到根的距离就是查找父亲需要的次数。

将小树并入大树,可以避免树的深度过深。

这个rnk数组保存的是树的深度的上界。

通常写法:

const int maxn = 10000000;
int rnk[maxn],fa[maxn];//rnk为该节点所在子树的深度
inline void unite(const int &x,const int &y)
{
//合并x与y的块
int t1 = find(x),t2 = find(y);//找到各自的祖宗
if(t1 == t2)
return;
if(rnk[t1] == rnk[t2])
{
//此时两树深度相等,随便合并
fa[t1] = t2;
++rnk[t2];//合并后深度增加
}
else if(rnk[t1] > rnk[t2])
fa[t2] = t1;
else
fa[t1] = t2;
}

总结

路径压缩和按秩合并都能将并查集复杂度降为O(logn)O(logn)O(logn),而两者一起使用能够降为O(log∗n)O(log^*n)O(log∗n)。

nnn log∗nlog^*nlog∗n
(−∞, 1] 0
(1, 2] 1
(2, 4] 2
(4, 16] 3
(16, 65536] 4
(65536, 2^65536] 5

(数据转自链接,个人认为这篇博文讲并查集讲得相当不错。)

代码

因为按秩合并需要额外数组,而在此数量级下开数组的消耗可能比优化更大……

于是借用了题目中维护的“连通块大小”,即节点个数,进行了按size合并的优化。

#include <cstdio>
using namespace std;
#define getId(x, y) (((x - 1) * n) + y) //将二维的点赋予一个一维的别名
int fa[1001000];
int h[1001000];
char all[2][1010];
int n, m, i, j;
//读入优化
inline char nc()
{
static char buf[400000], *p1 = buf, *p2 = buf;
return p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 400000, stdin), p1 == p2) ? EOF : *p1++;
}
void read(char *s)
{
static char c;
for (c = nc(); c != '1' && c != '0'; c = nc());
for (; c == '0' || c == '1'; *++s = c, c = nc());
}
void read(int &r)
{
static char c; r = 0;
for (c = nc(); c > '9' || c < '0'; c = nc());
for (; c >= '0' && c <= '9'; r = (r << 1) + (r << 3) + (c ^ 48), c = nc());
}
//并查集
inline int find(const int &x)
{
return fa[x] == x ? x : fa[x] = find(fa[x]);//路径压缩
}
inline void unite(const int &a, const int &b)
{
int t1 = find(a), t2 = find(b);
if (t1 == t2)
return;
h[t1] > h[t2] ? fa[t2] = t1, h[t1] += h[t2] : fa[t1] = t2, h[t2] += h[t1]; //类似按秩合并的做法,合并同时将儿子的大小加入父亲的大小中
}
//
int main()
{
read(n);
read(m);
int n2 = n * n;
for (int i = 1; i <= n2; ++i)
fa[i] = i, h[i] = 1; //初始化每个点都是单独的联通块,大小为1 int now = 0, pre = 1;
//第一行特判,不需要与上一行作比较
i = 1;
read(all[now]);
for (j = 2; j <= n; ++j)//j从2开始,可以略过第一个格子,最左上角的格子无需判断
if (all[now][j] != all[now][j - 1])
unite(getId(i, j), getId(i, j - 1)); for (i = 2; i <= n; ++i)
{
now ^= 1;//使用滚动数组
pre ^= 1;
read(all[now]);
if (all[now][1] != all[pre][1])
unite(getId(i, 1), getId(i - 1, 1));//第一个格子无需与左边判断
for (j = 2; j <= n; ++j)
{
if (all[now][j] != all[now][j - 1]) //因为是按行按列遍历,实际上只需要处理每个点的上方点和左边点的合并
unite(getId(i, j), getId(i, j - 1));
if (all[now][j] != all[pre][j])
unite(getId(i, j), getId(i - 1, j));
}
}
int x, y;
for (i = 1; i <= m; ++i)
{
read(x);
read(y);
printf("%d\n", h[find(getId(x,y))]);
}
return 0;
}

[洛谷Luogu]P1141 01迷宫[联通块 并查集]的更多相关文章

  1. luogu P1141 01迷宫 x

    P1141 01迷宫 题目描述 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上. 你的任 ...

  2. luogu P1141 01迷宫

    https://www.luogu.org/problem/show?pid=1141 还不太会用 BFS 然后就跟着感觉走了一波 经历了很多错误 刚开始的读入 然后BFS的过程 最后T三个点 看到别 ...

  3. Luogu P1141 01迷宫【搜索/dfs】By cellur925

    题目传送门 我tm到现在还需要刷这种水搜索...我退役吧. 但就是搜索弱嘛 补一补嘛qwq 题目大意:给你一张地图与许多询问,每次询问求这个点所在联通块的点的个数. 所以这个题目的本质就是在求联通块. ...

  4. 洛谷P1991无线通讯网[kruskal | 二分答案 并查集]

    题目描述 国防部计划用无线网络连接若干个边防哨所.2 种不同的通讯技术用来搭建无线网络: 每个边防哨所都要配备无线电收发器:有一些哨所还可以增配卫星电话. 任意两个配备了一条卫星电话线路的哨所(两边都 ...

  5. 洛谷P4768 [NOI2018]归程(可持久化并查集,最短路)

    闲话 一个蒟蒻,在网络同步赛上进行了这样的表演-- T2组合计数不会,T3字符串数据结构不会,于是爆肝T1 一开始以为整个地图都有车,然后写了2h+的树套树,终于发现样例过不去 然后写可持久化并查集D ...

  6. 洛谷P3402 【模板】可持久化并查集 [主席树,并查集]

    题目传送门 可持久化并查集 n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0 ...

  7. 洛谷 P2024 食物链 POJ 1182 Label:并查集Turbo

    题目描述 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B 吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的一种,但是我 ...

  8. 洛谷P3402 【模板】可持久化并查集(可持久化线段树,线段树)

    orz TPLY 巨佬,题解讲的挺好的. 这里重点梳理一下思路,做一个小小的补充吧. 写可持久化线段树,叶子节点维护每个位置的fa,利用每次只更新一个节点的特性,每次插入\(logN\)个节点,这一部 ...

  9. [洛谷P2024/POJ1182]食物链 - 带偏移量的并查集(2)

    Description 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的 ...

随机推荐

  1. Codeforces1303C. Perfect Keyboard

    本题可以转换成图论来做,每两个相邻点连一条边,然后统计每一个点的degree,如果>=2说明有一个点要相邻三个点,不满足题意,然后找出每个degree<2的点,这些点可以作为一段的起点,然 ...

  2. ImageSwitcher和GridView的案例开发

    (一)ImageSwitcher之手机相册的滑动查看 首先在布局文件上加一个ImageSwitcher,设置它的宽度和高度为match_parent. 在主程序中:首先设置一个存储照片资源的数组,在设 ...

  3. 单表千亿电信大数据场景,使用Spark+CarbonData替换Impala案例

    [背景介绍] 国内某移动局点使用Impala组件处理电信业务详单,每天处理约100TB左右详单,详单表记录每天大于百亿级别,在使用impala过程中存在以下问题: 详单采用Parquet格式存储,数据 ...

  4. ping命令工具:同时ping多个IP

    检测多个ip在同一时间点的响应状态,通过对比来判断哪个ip异常. 下载地址:https://share.weiyun.com/5XCkypG

  5. CentOS安装后的第一步:配置IP地址

    有关于centos7获取IP地址的方法主要有两种,1:动态获取ip:2:设置静态IP地址 在配置网络之前我们先要知道centos的网卡名称是什么,centos7不再使用ifconfig命令,可通过命令 ...

  6. JAVA 数据库操作工具类----sqllite

    package com.asc.db; import android.content.ContentValues; import android.content.Context; import and ...

  7. Vue.js 内联样式绑定style

    html <div class="Menu" v-bind:style="{height:clientHeight}"> </div> ...

  8. re模块补充 configparse模块

    import rere.findall("(?:abc)+","abcabcabc")--->['abcabcabc'] import configpar ...

  9. Windows篇:文件对比软件->"DiffMerge"

    文件对比软件->"DiffMerge" DiffMerge是什么? 如果没有DiffMerge! 想想一下,有两篇10000字的文章,找不同,眼睛都要看花吧.有了DiffMe ...

  10. RCast 66: 射影几何与Rho演算

    Greg Meredith与Isaac DeFrain和Christian Williams一起讨论了射影几何及其在Rho演算中的作用. 原文链接及音频 https://blog.rchain.coo ...