POJ 1182 食物链(并查集的使用)
食物链
| Time Limit: 1000MS | Memory Limit: 10000K | |
| Total Submissions: 81915 | Accepted: 24462 |
Description
现有N个动物,以1-N编号。每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种。
有人用两种说法对这N个动物所构成的食物链关系进行描述:
第一种说法是"1 X Y",表示X和Y是同类。
第二种说法是"2 X Y",表示X吃Y。
此人对N个动物,用上述两种说法,一句接一句地说出K句话,这K句话有的是真的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。
1) 当前的话与前面的某些真的话冲突,就是假话;
2) 当前的话中X或Y比N大,就是假话;
3) 当前的话表示X吃X,就是假话。
你的任务是根据给定的N(1 <= N <= 50,000)和K句话(0 <= K <= 100,000),输出假话的总数。
Input
以下K行每行是三个正整数 D,X,Y,两数之间用一个空格隔开,其中D表示说法的种类。
若D=1,则表示X和Y是同类。
若D=2,则表示X吃Y。
Output
Sample Input
100 7
1 101 1
2 1 2
2 2 3
2 3 3
1 1 3
2 3 1
1 5 5
Sample Output
3 对于每只动物i创建3个元素i-A,i-B,i-C,并用3×N个元素建立并查集,维护如下信息:
1、i-X表示i属于X
2、如果i-A和j-B在同一个组里,那么如果i属于A,j就一定属于B。如果j属于B,i就一定属于A。
我们对每条信息进行如下操作:
1、如果x或者y比N大或者小于1,则答案+1
2、x和y属于同一种类。如果已知x和y不属于同一类,则答案+1。否则合并x-A和y-A、x-B和y-B、x-C和y-C
3、x吃y。如果已知x和y属于同一类或者x-A和y-C在同一个集合,则答案+1。否则合并x-A和y-B、x-B和y-C、x-C和y-A
最后,输出答案。
在第一次交的时候我的程序超时了,我以为是cin的问题,所以改为了scanf,但依然超时。之后我才想到了需要用启发式合并(就是将高度较小的树合并在高度较大的树下面),不然时间不够。 当然,查找时要带路径压缩。
代码2L
由于N和K 很大,所以必须高效地维护动物之间关系,并快速判断是否产生了矛盾。并查集是维护“属于同一组”的数据结构,但是本题中,并不是只有属于同一类的信息,还有捕食关系的存在。因此需要思考如何维护这些关系。
对于每只动物i创建3个元素 i - A, i - B, i - C, 并用这3 x N个元素建立并查集。这个并查集维护如下信息:
i-x表示 “i属于种类x"
并查集里每个组表示组内所有元素代表的情况都同时发生或不发生。
例如:如果i-A和j-B在同一个组里,就表示如果i属于种类A那么j一定属于种类B,如果i属于种类B那么j一定属于种类A。因此,得出下面的操作。
第一种,x和y属于同一种类·······合并x-A和y-A、x-B和y-B、x-C和y-C。
第二种,x吃y······························合并x-A和y-B、x-B和y-C、x-C和y-A。
不过在合并前,需要先判断合并是否会产生矛盾。例如在第一种信息的情况下,需要检查(x-A, y-B) ||(x-A, y-C)是否在同一组等。
#include <iostream>
#include <string>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = ;
int n, k;
int d, x, y;
int ans = ;
int par[maxn];
int Rank[maxn];
void init(int n)//初始化
{
int i;
for (i = ; i <= * n; i++)
{
par[i] = i;
Rank[i] = ;
}
}
int find(int x)
{
if (par[x] == x) return x;
else return par[x]=find(par[x]);//找到就直接连到树的跟上(因为它超时多次)
}
bool same(int x, int y)
{
return find(x) == find(y);
}
void unite(int xx, int yy)//合并
{
int x = find(xx);//局部变量,WA了好多次
int y = find(yy);
if (x == y) return;
if (Rank[x] < Rank[y]) par[x] = y;//树矮的放到高的那里
else
{
par[y] = x;
if (Rank[x] == Rank[y]) Rank[x]++;//两棵树同样高时,合并要高度++
}
}
void solve()
{
if (x <= || x > n || y <= || y > n)
{
ans++; return;
}
if (d == )
{
// "x和y属于同一类"的信息
if (same(x, y + n) || same(x, y + * n))
{
ans++; }
else
{
unite(x, y);
unite(x + n, y + n);
unite(x + n * , y + n* );
}
}
else
{
//"x吃y"的信息
if (same(x, y) || same(x, y + * n))
{ //A和A,A和C不能相等
ans++; }
else
{
unite(x, y + n);
unite(x + n, y + * n);
unite(x + * n, y);
}
}
}
int main()
{
scanf("%d %d", &n, &k);
init(n);
while (k--)
{
scanf("%d %d %d", &d, &x, &y);
solve();
}
printf("%d\n", ans);
return ;
}
精彩的代码
POJ 1182 食物链(并查集的使用)的更多相关文章
- POJ 1182 食物链 [并查集 带权并查集 开拓思路]
传送门 P - 食物链 Time Limit:1000MS Memory Limit:10000KB 64bit IO Format:%I64d & %I64u Submit ...
- poj 1182 食物链 并查集 题解《挑战程序设计竞赛》
地址 http://poj.org/problem?id=1182 题解 可以考虑使用并查集解决 但是并不是简单的记录是否同一组的这般使用 每个动物都有三个并查集 自己 天敌 捕食 并查集 那么在获得 ...
- poj 1182 食物链 并查集的又一个用法
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 41584 Accepted: 12090 Descripti ...
- POJ 1182食物链(并查集)
食物链Time Limit: 1000MS Memory Limit: 10000KTotal Submissions: 85474 Accepted: 25549Description动物王国中有三 ...
- POJ - 1182 食物链 并查集经典
思路:设r(x)表示节点x与根结点的关系,px表示x的根结点.记录每个节点与其父节点的关系,就能很方便知道每个节点以及和它的父节点的关系. struct node{ int par; //父亲节点 i ...
- poj——1182食物链 并查集(提升版)
因为是中文题,题意就不说了,直接说思路: 我们不知道给的说法中的动物属于A B C哪一类,所以我们可以用不同区间的数字表示这几类动物,这并不影响结果,我们可以用并查集把属于一类的动物放在一块,举个例子 ...
- POJ 1182 食物链 (并查集)
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 50601 Accepted: 14786 Description ...
- POJ 1182 食物链(并查集)
题目链接 经过宝哥的讲解,终于对这种问题有了进一步的理解.根据flag[x]和flag[y]求flag[tx]是最关键的了. 0吃1,1吃2,2吃0. 假设flag[tx] = X; 那么X + fl ...
- poj 1182 (关系并查集) 食物链
题目传送门:http://poj.org/problem?id=1182 这是一道关系型并查集的题,对于每个动物来说,只有三种情况:同类,吃与被吃: 所以可以用0,1,2三个数字代表三种情况,在使用并 ...
随机推荐
- 对小波变换中DWT和CWT的理解
转载自 http://blog.sina.com.cn/s/blog_633750d90100hbco.html 连续小波变换的概念.操作.及时间尺度图的显示 最近很多网友问到关于连续小波变换的诸多问 ...
- JVM自动内存管理:对象判定和回收算法
可回收对象的判断方法 1.引用计数算法 2.可达性分析算法 引用计数算法 给对象中添加一个引用计数器,每当有一个地方引用它时,计数器值就加1:当引用失效时,计数器值就减1:任何时刻计数器为0的对象就是 ...
- cobbler网络装机
cobbler网络装机 原理分析 cobbler简介 Cobbler通过将设置和管理一个安装服务器所涉及的任务集中在一起,从而简化了系统配置.相当于Cobbler封装了DHCP.TFTP.XINTED ...
- select 从应用层到内核实现解析
在一个应用中,如果需要读取多个设备文件,这其中有多种实现方式: 1.使用一个进程,并采用同步查询机制,不停的去轮询每一个设备描述符,当设备描述符不可用时,进程睡眠. 2:使用多个进程或者线程分别读取一 ...
- file.replace
一.简介 salt file.replace 文件内容处理函数,类似于ansible的lineinfile模块 二.参数介绍 name 被编辑文件的绝对路径,支持软链接 pattern 常规表达式,使 ...
- removeLineEndSpace
/****************************************************************************** * removeLineEndSpace ...
- matrix-gui-browser-2.0 matrix-browser Qt QWebView hacking
/* * matrix-browser * * Simple web viewer used by Matrix application launcher * * Copyright (C) 2011 ...
- shell 脚本实战笔记(4)--linux磁盘分区重新挂载
背景: Hadoop的HDFS文件系统的挂载, 默认指定的文件目录是/mnt/disk{N}. 当运维人员, 不小心把磁盘挂载于其他目录, 比如/mnt/data, /mnt/disk01, /mnt ...
- pymysql中如何将动态的插入数据库中
data = { ', 'name': 'zengsf', 'age': 20 } table = 'students' #获取到一个以键且为逗号分隔的字符串,返回一个字符串 keys = ', '. ...
- each遍历 的原理
<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>& ...