题目链接:http://poj.org/problem?id=1182

食物链
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 78133   Accepted: 23275

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

带权并查集:

 
  带权:r[]数组可以记录当前结点与父节点的关系,可以是大小关系, 可以是逻辑关系(如此题)。对于相同的集合,由于在这棵树中,每个结点与父节点的关系已经确定,那么每个节点与集合中的其他结点的关系也可以一路推导出来。对于两个不同的集合,如果知道一对位于不同集合的结点的关系,那么这两个集合所有的结点之间的关系也可以推导出来了,即两个集合可以合并为一个集合。  
 
  路径压缩:对于被find()函数访问过的结点x, 它们的fa[x]都会直接指向根节点,同时需要更新r[x]数组(一路叠加)。问:那么对于被访问过的结点x的子树怎么办呢,不会被落下吗?答:结点x的子树的fa[]指针没有改变,仍然是指着x,即x的子树一直跟着x。
 
  合并:对于两个不同的集合,由于在对u、v调用find()函数时,u和v都分别指向了各自的根节点(路径压缩)。设fu为u所在集合的根节点(也是u的父节点), fv也如此,所以u和fu的关系即为r[u]、v和fv的关系即为r[v],且又知道u和v的关系, 那么就可以直接推出fu和fv的关系,这样就可以实现两个集合的合并。
 

代码如下:

 #include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <stack>
#include <map>
#include <string>
#include <set>
#define ms(a,b) memset((a),(b),sizeof((a)))
using namespace std;
typedef long long LL;
const double EPS = 1e-;
const int INF = 2e9;
const LL LNF = 2e18;
const int MAXN = 5e4+; int n, m;
int fa[MAXN], r[MAXN]; int find(int x)
{
if(fa[x]==-) return x;
int pre = find(fa[x]);
r[x] = (r[x]+r[fa[x]])%;
return fa[x] = pre;
} bool Union(int w, int u, int v)
{
int fu = find(u);
int fv = find(v);
if(fu==fv)
return ((-w+r[u])%!=r[v]); fa[fu] = fv;
r[fu] = (-r[u]+w+r[v])%;
return false;
} int main()
{
scanf("%d%d", &n, &m);
memset(r, , sizeof(r));
memset(fa, -, sizeof(fa)); int ans = ;
for(int i = ; i<=m; i++)
{
int d, u, v;
scanf("%d%d%d", &d, &u, &v);
if(u>n || v>n)
ans++;
else if(d== && u==v)
ans++;
else if(Union(d-, u, v))
ans++;
}
printf("%d\n", ans);
}

最新文章

  1. allow zero datetime=true导致datetime转换失败:MySql.Data.Types.MySqlDateTime”的对象无法转换为类型“System.Nullable`1[System.DateTime]
  2. Lua系统库
  3. JS - To my gril
  4. CF160D
  5. 使用DX绘制3D物体时新手常犯错误,看不见物体时可以一一排查
  6. MSSQL 2005数据库与SP4补丁安装
  7. 小白日记9:kali渗透测试之主动信息收集(二)四层发现:TCP、UDP、nmap、hping、scapy
  8. hdu2569(递推dp)
  9. 丘吉尔,最后一次演讲-Never Give Up
  10. Web前端相关资源
  11. Mac环境下实现alias重命名命令(永久生效)
  12. MySQL数据库学习: 02 —— 数据库的安装与配置
  13. Win7如何解决telnet不是内部或外部命令的方案!听语音
  14. Java 中数字和字符串拼接的问题
  15. JS学习笔记Day17
  16. ::class 意思
  17. [PC]两个蓝牙耳机同时输出相同音源
  18. 【iCore4 双核心板_FPGA】例程五:基础逻辑门实验——逻辑门使用
  19. 【netcore入门】在Windows IIS上部署.NET Core 2.1项目
  20. #include&lt;iostream&gt;与#include&lt;iostream.h&gt;以及#inclue&lt;string&gt; 和 #include&lt;string.h&gt;的区别

热门文章

  1. 【06】Firebug记录Javascript日志
  2. 关于构造函数什么值传递给他的实例,只有this和prototype
  3. 使用imageMagick 制作圆角矩形和图片加水印
  4. 2018-2019 XIX Open Cup, Grand Prix of Korea (Division 2) GYM 102058 F SG函数
  5. 2018 ACM 国际大学生程序设计竞赛上海大都会赛重现赛 A,D
  6. ztr loves lucky numbers--hdu5676(DFS)
  7. 寒武纪camp Day4
  8. datasnap中间件如何控制长连接的客户端连接?
  9. .NET作品集:基于svn 的.net 持续集成工具
  10. IOS开发 APP提交程序上传流程