食物链
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 82346   Accepted: 24616

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

思路:

因为要表达并查集间的关系,所以我们设了一组数组rela[i]来表示i与他的根之间的关系(即表示i到pre[i]的偏移量),假设a b,0表示同类,1表示a吃b,2表示b吃a。

在压缩路径时,我们需要对沿途的数据更新rela:rela[a]=(rela[a]+rela[ pre[a] ])%3;  此代码应这样理解:rela表示的是a对当前的根aa的偏移量,即a->aa,而aa对最顶端的根fa偏移量为rela[ aa ],即aa->fa,所以将两者相加(注意!这是个向量,即a->根)就为a对最顶端的根fa偏移量,即a->fa。

在执行join时我们注意到我们需要更新合并后fb的rela,代码为:rela[fb]=(3-rela[b]+(c-1)+rela[a])%3;  要了解这个,首先我们要知道一个知识:

假如x作为y的根时,y的偏移量是a(即y->x),那么y作为x的父亲时,x的偏移量就是(3 - a)%3(即x->y)

所以代码应这样理解:(3-rela[b])是fb对b的偏移量即fb->b, (c-1)是b对a偏移量即b->a,(rela[a]) 是a对fa的原根偏移量

即a->fa ,所以fb->b +b->a + a->fa =fb->fa,即为fb对fa偏移量。

我也是在看了大佬详细解说之后才恍然大悟,特奉上参考博客:点此

代码:

#include<stdio.h>
#include<string.h>
#include<math.h>
#include<stdlib.h>
#include<map>
#define N 50010
#define HASH 10000
using namespace std;
int pre[N],rela[N],c; int find(int a){
if(pre[a]==a) return a;
else{
int temp=pre[a];
pre[a]=find(temp);
rela[a]=(rela[a]+rela[temp])%3; //当前a对根的偏移量
return pre[a];
}
}
void join(int a,int b){
int fa=find(a);
int fb=find(b);
if(fa!=fb){
pre[fb]=fa;
rela[fb]=(3-rela[b]+(c-1)+rela[a])%3;//(3-rela[b])是b与原根fb的偏移量, (c-1)是a b间偏移量,(rela[a]) 是a和它的原根fa偏移量 ,三者相加为fa fb间偏移量
}
}
int main(){
int n,i,k,answer=0,a,b;
scanf("%d%d",&n,&k);
for(i=0;i<=n;i++){
pre[i]=i;
rela[i]=0;
}
while(k--){
scanf("%d%d%d",&c,&a,&b);
if(c==2 && a==b) answer++;
else if(a>n ||b>n) answer++;
else if(find(a)!=find(b)){
join(a,b);
}
else if(c==1){
if(rela[a]!=rela[b])
answer++;
}
else{
if( ((rela[b]+3-rela[a])%3)!=1 ) //代表a为b根时b的偏移量
answer++;
}
}
printf("%d\n",answer);
return 0;
}

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: 81915   Accepted: 24462 Description ...

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

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

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

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

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

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

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

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

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

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

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

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

随机推荐

  1. html5 随机数函数

    function selec(low,high){var ch=high-low+1;return Math.floor(Math.random()*ch+low);}for (var i = 0; ...

  2. 获取MyBatis

    点击:获取MyBatis https://github.com/mybatis/mybatis-3/releases 点击:进入中文MyBatis的说明文档 http://www.mybatis.or ...

  3. uva 1633 Dyslexic Gollum

    题意: 给出n和k,求出长度为n的不包含长度大于等于k的回文串的01字符串的个数. 思路: 如果一个字符串包含长度为k的回文串,那么它肯定包含长度为k-1的回文串,所以考虑第i位的时候,只要前缀中不包 ...

  4. Python学习记录之----网络通信(二)

    网络通信   socket 这一节太难了,还是看TA的吧 http://www.cnblogs.com/alex3714/articles/5830365.html 不能执行top等类似的 会持续输出 ...

  5. 第三方包源码maven 下载

    1,maven导包时候自动下载源码 在eclipse或者Myeclipse中 => window ==> preference ==> Maven ==> 勾选Download ...

  6. spring4.0.0 源码导入eclipse(sts)

    其余步骤请见:http://www.cnblogs.com/xiluhua/p/7450972.html 执行 gradle eclipse -x :eclipse 报错: 解决办法: 找到 行,注释 ...

  7. CBV源码解析

    1.CBV(class based view) 首先定义一个视图函数,用类的方式定义: 举例: class LoginView(View): def get(self,request): return ...

  8. PCB 布线,直角线,差分线,蛇形线

    1.直角线 直角走线的一般标准是PCB布线中要尽量避免的情况,也几乎成为衡量布线好坏的标准之一. 直角走线对信号的影响主要体系那在下面三个方面 1.保教可以等效为传输线是哪个的容性负载,减缓上升时间. ...

  9. RESTful API 设计指南,RESTful API 设计最佳实践

    RESTful API 设计指南,RESTful API 设计最佳实践 网络应用程序,分为前端和后端两个部分.当前的发展趋势,就是前端设备层出不穷(手机.平板.桌面电脑.其他专用设备......). ...

  10. Java中的ASCII码与Unicode码

    先上代码 后续更新 public class Unicode { public static void main(String[] args) { char ch1 = 'c'; char ch2 = ...