poj_1988 并查集
题目大意
开始有N堆砖块,编号为1,2....N,每堆都只有一个。之后可以进行两种操作:
(1)M X Y 将编号为X的砖块所在的那堆砖拿起来放到编号为Y的砖块所在的堆上;
(2)C X 查询编号为X的砖块所在的堆中,在砖块X下方的所有砖块的数目
题目分析
典型的集合合并和查询,因此采用并查集。并查集的基本框架就是一个GetPar函数(实现查找集合的祖先,同时实现路径压缩),一个Union函数(实现将两个集合合并),一个SameGroup函数(判断两个元素是否属于同一个集合)。
在利用并查集解决具体问题的时候,需要做的是设置一个数据结构用于存放问题所需要的信息,然后在GetPar函数和Union函数中更新这个数据结构。
在本题中,维护信息 SumOfStack(每堆中的所有砖块数目),NumOfUnderBlock(堆中在砖块下方的砖块数目)。将每堆砖块集合的编号(即集合的根)设置为该堆最下方的砖块号,则在合并的时候,上堆的根节点的父节点设置为下堆的根节点,可以更新的数据为上堆的根节点下方的砖块数目和下堆的根节点代表的堆总砖块数。
这样,每堆的根节点的SumOfStack信息是正确的,而每次合并后上堆的原根节点的NumOfUnderBlock信息也是正确的;对于某个砖块b,在GetPar的过程中,由于b的gPar[b]可能没被更新,如果没被更新,则b的NumOfUnderBlock[b]表示在b之前所在的堆中位于b下方的砖块数目也是正确的,于是将此时的NumOfUnderBlock[b] + NumOfUnderBlock[gPar[b]](注意,这里的gPar[b]为b原来的堆中的根),即可得到最终的NumOfUnderBlock[b],这可以在GetPar函数的递归中完成。
实现(c++)
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#define MAX_STACK_NUM 30001
int gPar[MAX_STACK_NUM];
int gSumOfStack[MAX_STACK_NUM];
int gNumOfUnderBlock[MAX_STACK_NUM]; int GetPar(int c){
if (c != gPar[c]){
int p = gPar[c];
gPar[c] = GetPar(gPar[c]); //信息维护
gNumOfUnderBlock[c] += gNumOfUnderBlock[p];
}
return gPar[c];
} void Union(int a, int b){
int p1 = GetPar(a);
int p2 = GetPar(b);
gPar[p1] = p2; //信息维护
gNumOfUnderBlock[p1] = gSumOfStack[p2];
gSumOfStack[p2] += gSumOfStack[p1];
} void Init(){
for (int i = 0; i < MAX_STACK_NUM; i++){
gSumOfStack[i] = 1;
gNumOfUnderBlock[i] = 0;
gPar[i] = i;
}
}
int main(){
int p, X, Y;
scanf("%d", &p);
char op;
Init();
for (int i = 0; i < p; i++){
getchar();
scanf("%c", &op);
if (op == 'M'){
scanf("%d %d", &X, &Y);
Union(X, Y);
}
else if (op == 'C'){
scanf("%d", &X);
GetPar(X);
printf("%d\n", gNumOfUnderBlock[X]);
}
}
return 0;
}
poj_1988 并查集的更多相关文章
- BZOJ 4199: [Noi2015]品酒大会 [后缀数组 带权并查集]
4199: [Noi2015]品酒大会 UOJ:http://uoj.ac/problem/131 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 ...
- 关押罪犯 and 食物链(并查集)
题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用"怨气值"( ...
- 图的生成树(森林)(克鲁斯卡尔Kruskal算法和普里姆Prim算法)、以及并查集的使用
图的连通性问题:无向图的连通分量和生成树,所有顶点均由边连接在一起,但不存在回路的图. 设图 G=(V, E) 是个连通图,当从图任一顶点出发遍历图G 时,将边集 E(G) 分成两个集合 T(G) 和 ...
- bzoj1854--并查集
这题有一种神奇的并查集做法. 将每种属性作为一个点,每种装备作为一条边,则可以得到如下结论: 1.如果一个有n个点的连通块有n-1条边,则我们可以满足这个连通块的n-1个点. 2.如果一个有n个点的连 ...
- [bzoj3673][可持久化并查集 by zky] (rope(可持久化数组)+并查集=可持久化并查集)
Description n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0 0& ...
- [bzoj3123][sdoi2013森林] (树上主席树+lca+并查集启发式合并+暴力重构森林)
Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...
- 【BZOJ-3673&3674】可持久化并查集 可持久化线段树 + 并查集
3673: 可持久化并查集 by zky Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 1878 Solved: 846[Submit][Status ...
- Codeforces 731C Socks 并查集
题目:http://codeforces.com/contest/731/problem/C 思路:并查集处理出哪几堆袜子是同一颜色的,对于每堆袜子求出出现最多颜色的次数,用这堆袜子的数目减去该值即为 ...
- “玲珑杯”ACM比赛 Round #7 B -- Capture(并查集+优先队列)
题意:初始时有个首都1,有n个操作 +V表示有一个新的城市连接到了V号城市 -V表示V号城市断开了连接,同时V的子城市也会断开连接 每次输出在每次操作后到首都1距离最远的城市编号,多个距离相同输出编号 ...
随机推荐
- 【C#】Config配置文件的读写,及无法写入/保存配置文件的问题
目的: 一些数据为了在项目打包好后也能方便的修改和调用,通常会把这些数据放到配置文件中,避免硬编码,修改配置文件内容更方便,而不用修改源代码. 使用: 在解决方案资源管理器中找到App.config文 ...
- JEECG常见问题大全征集
大家还有什么问题.请跟帖,谢谢支持. . JEECG常见问题大全征集 1. jeecg没有数据库脚本问题 jeecg不须要数据库脚本,在数据库创建好数据库.项目配置好数据源链接.会自己主动建表. ...
- copy src remote_src false表示本地,true在远程
文件组装模块-assemble assemble主要是将多份配置文件组装为一份配置文件. 参数 必填 默认 选项 说明 Backup 否 No Yes/no 是否创建备份文件,使用时间戳 Delimi ...
- jQuery Form插件详解
<script src="js/jquery.form.js" type="text/javascript"></script> Jqu ...
- (转)Invalidate、RedrawWindow与UpdateWindow的区别
一:什么时候才会发生重绘窗口的消息? 当需要更新或重新绘制窗口的外观时,应用程序就会发送WM_PAINT消息.对窗口进行重新绘制. 二:Invalidate() -- RedrawWindow() ...
- python 脚本检测python 版本
通过sys 模块的sys_info可以返回当前python 的版本信息, 其返回值是一个元组, 比如(2, 6, 6, 'final', 0); 表示当前版本为2.6.6 , 我们可以利用这个变量的值 ...
- CentOS安装emacs24.2命令
CentOS安装emacs24.2命令 #1.安装如下软件 yum -y groupinstall "Development Tools" yum -y install gtk+- ...
- 【Java面试题】38 Collection 和 Collections的区别
Collection是集合类的一个顶级接口,其直接继承接口有List与Set 而Collections则是集合类的一个工具类/帮助类,其中提供了一系列静态方法,用于对集合中元素进行排序.搜索以及线程安 ...
- htaccess正则规则学习笔记整理
# —— 位于行首时表示注释. [F] —— Forbidden(禁止): 命令服务器返回 403 Forbidden错误给用户浏览器 [L] —— Last rule(最后一条规则): 告诉服务器在 ...
- Javascript的setTimeOut和setInterval的定时器用法
setTimeout() 方法用于在指定的毫秒数后调用函数或计算表达式,而setInterval()则是在每隔指定的毫秒数循环调用函数或表达式, 直到 clearInterval把它清除.也就是说se ...