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三个数字代表三种情况,在使用并 ...
随机推荐
- DevExpress v18.1新版亮点——WinForms篇(七)
用户界面套包DevExpress v18.1日前终于正式发布,本站将以连载的形式为大家介绍各版本新增内容.本文将介绍了DevExpress WinForms v18.1 的新功能,快来下载试用新版本! ...
- DevExpress v17.2新版亮点—DevExtreme篇(二)
用户界面套包DevExpress DevExtreme v17.2终于正式发布,本站将以连载的形式为大家介绍各版本新增内容.本文将介绍了DevExtreme v17.2 的New Color Sche ...
- SQL 递归找查所有子节点及所有父节
在SQL的树型结构中,很多时候,知道某一节点的值,需要查找该节点的所有子节点(包括多级)的功能,这时就需要用到如下的用户自定义函数. 表结构如下: ID int Dep_Type int Dep_Co ...
- 获取bean的两种方式
BeanFactory方式: 1: public void testFactory(){ ResourcePatternResolver rpt=new PathMatchingResourcePat ...
- 在 windows 开发 reactNative 的环境 搭建过程 react-native-android
安装的东西挺多的, 从 jdk 到c++环境 到node , python, 各种模拟器 http://bbs.reactnative.cn/topic/10/%E5%9C%A8windows%E4% ...
- Java实现交换两个String
在Java中我们所使用的实例变量其实都是一个引用,所以如果要求实现一个swap(String A, String B)这种函数时无法实现的,因为在类方法的定义中是先对行参进行地址传递,然后对形参修改, ...
- react 学习日记
1.本地配置代理服务: create-react-app 创建的react项目 package.jsoin 中 加入: "proxy": "http://localh ...
- conan-transit服上的库列表
conan-transit服上的库列表 因为获取列表比较慢,所以获取后在此记录,以备查找.conan-transit 是个只读库,不会有更新.新的库将上传到 conan-center. conan是C ...
- Android AES 加密、解密
AES加密介绍 ASE 加密.解密的关键在于秘钥.只有使用加密时使用的秘钥,才可以解密. 生成秘钥的代码网上一大堆,下面的代码可生成一个秘钥 private SecretKey generateKey ...
- cache和buffer区别
Cache: 一般用于读缓存,用于将频繁读取的内容放入缓存,下次在读取相同的内容,直接从缓存冲读取,提高读取性能,缓存可以有多级. Buffer:一般用于写缓存,用于解决不同介质直接存储速度的不同, ...