食物链

Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 81915   Accepted: 24462

Description

动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形。A吃B, B吃C,C吃A。 
现有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

第一行是两个整数N和K,以一个空格分隔。 
以下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 食物链(并查集的使用)的更多相关文章

  1. POJ 1182 食物链 [并查集 带权并查集 开拓思路]

    传送门 P - 食物链 Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%I64d & %I64u Submit  ...

  2. poj 1182 食物链 并查集 题解《挑战程序设计竞赛》

    地址 http://poj.org/problem?id=1182 题解 可以考虑使用并查集解决 但是并不是简单的记录是否同一组的这般使用 每个动物都有三个并查集 自己 天敌 捕食 并查集 那么在获得 ...

  3. poj 1182 食物链 并查集的又一个用法

    食物链   Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 41584   Accepted: 12090 Descripti ...

  4. POJ 1182食物链(并查集)

    食物链Time Limit: 1000MS Memory Limit: 10000KTotal Submissions: 85474 Accepted: 25549Description动物王国中有三 ...

  5. POJ - 1182 食物链 并查集经典

    思路:设r(x)表示节点x与根结点的关系,px表示x的根结点.记录每个节点与其父节点的关系,就能很方便知道每个节点以及和它的父节点的关系. struct node{ int par; //父亲节点 i ...

  6. poj——1182食物链 并查集(提升版)

    因为是中文题,题意就不说了,直接说思路: 我们不知道给的说法中的动物属于A B C哪一类,所以我们可以用不同区间的数字表示这几类动物,这并不影响结果,我们可以用并查集把属于一类的动物放在一块,举个例子 ...

  7. POJ 1182 食物链 (并查集)

    食物链 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 50601   Accepted: 14786 Description ...

  8. POJ 1182 食物链(并查集)

    题目链接 经过宝哥的讲解,终于对这种问题有了进一步的理解.根据flag[x]和flag[y]求flag[tx]是最关键的了. 0吃1,1吃2,2吃0. 假设flag[tx] = X; 那么X + fl ...

  9. poj 1182 (关系并查集) 食物链

    题目传送门:http://poj.org/problem?id=1182 这是一道关系型并查集的题,对于每个动物来说,只有三种情况:同类,吃与被吃: 所以可以用0,1,2三个数字代表三种情况,在使用并 ...

随机推荐

  1. C++ error C2440: “类型转换” : 无法从“std::vector::iterator”转换为“

    原文地址:http://blog.csdn.net/onlyou930/article/details/5602654 圆环套圆环之迭代器 话说这一日是风平浪静,万里乌云,俺的心情好的没得说,收到命令 ...

  2. MyEclipse移动开发教程:迁移HTML5移动项目到PhoneGap(二)

    MyEclipse开年钜惠 在线购买低至75折!立即开抢>> [MyEclipse最新版下载] 二.将文件从HTML5项目复制到PhoneGap项目中 1. 在HTML5 app项目的ww ...

  3. 利用Teensy进行em410x卡模拟以及暴力破解em410x类门禁系统

    什么是低频?什么是EM410x? 首先,我不得不再次提一下那些工作在125khz频率下的低频卡(如:EM410X之类的),以便大家更好的阅读以下的内容. 什么是低频?以下就是低频的解释: 低频(LF, ...

  4. golang多进程测试代码

    package main import ( "fmt" "runtime" ) func test(c chan bool, n int) { x := 0 f ...

  5. magento小常识

    magento后台设置好产品分类及添加产品后前台没有显示出来:以下几个可能: 1.首先检查网店商城的Root Category 以 及跟目录下面的子目录设置是否有误,具体在目录->管理分类-&g ...

  6. preprocessor设置调试宏

    调试宏:preprocessor设置 预处理器“调试”宏在Xcode项目模板的调试版本定义.预处理宏在编译时被解释和调试宏可以用来允许调试代码运行在调试版本中你的项目.如果你不确定你的项目已经确定,可 ...

  7. 4天赚4000RMB 引流秘籍 卑鄙无耻之徒

    他说:有什么引流秘籍吗? 我说:贴吧.天涯.论坛都不错,只要不明着发广告,每天被动加几十个人轻轻松松的,如果执行力好,几百上千都有可能. 他说:我该怎么操作. 我说:网上找点好的帖子,简单修改一下部分 ...

  8. Ubuntu忘记root密码的解决方法

    如果是Linux操作系统的话,其实也是很简单 -- 单用户登陆.下面以Ubuntu14.04来简单演示一下具体的操作流程. 1. 开机 2. 此时会有一个选项:Advanced Options for ...

  9. CodeForces - 367E:Sereja and Intervals(组合数&&DP)

    Sereja is interested in intervals of numbers, so he has prepared a problem about intervals for you. ...

  10. MVC中未能加载程序集System.Web.Http/System.Web.Http.WebHost

    ==================================== 需要检查项目的Microsoft.AspNet.WebApi版本是否最新,System.Web.Http 这个命名空间需要更新 ...