食物链
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 71361   Accepted: 21131

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
欸困惑许久的一道题,寒假学并查集时已经遇见,卡在带权并查集一直没时间or感觉很复杂就没学,前几日突然想起,便想起他了.
带权并查集感觉在于对向量偏移的理解和对getf()函数递归查找祖先的深刻理解,怎奈数学砸砸看见数学就头疼>_<
本题体面给出了三个类,并查集的作用:将具有相同性质的某些点归到一颗树里,他们有共同的祖先.
难不成三个类要用三个并查集?显然用三个只会更加复杂,我们不妨利用带权并查集点到祖先的距离的不同来表示点与祖先的关系,将所有的点归到一颗树下,假设有两个点x,y在一颗树里且我们知道其到祖先的距离d[x],d[y],
显然利用这两个"距离"(关系)我们足以推出x,y的关系,这便涉及到向量了.
带权并查集主要就是在getf()函数递归的过程中顺便改变到根节点的距离这一步!合并函数中也需要进行小小的改变.
推论:已知x,x的父亲t=f[x],根祖先root和d[x],d[t]; 我们要做的就是根据 d[x],d[t]更新出当前正确的d[x]的值.
穷举出所有的可能后:{0,1,2}--{0,1,2}不难得出结论 d[x]=(d[x]+d[t])%3;
为了保证 d[t]的值是x的父亲到祖先的距离我们需要先递归x的父亲更新他父亲的距离......
如果不先进行递归的话,d[t]的值可能并不是t到其根的距离导致结果错误!
merg函数类似利用向量推出公式即可,利用0,1,2的巧妙之处在于可以用(d-1)直接表示出x与y的关系.

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int f[50005],d[50005];
int getf(int v)
{
if(f[v]==v) return v;
int t=f[v];
f[v]=getf(f[v]);
d[v]=(d[v]+d[t])%3;
return f[v];
}
void merg(int x,int y,int rx,int ry,int D)
{
f[ry]=rx;
d[ry]=(3-d[y]+(D-1)+d[x])%3;
}
int main()
{
int n,k,x,y,D,ans=0,i,j;
cin>>n>>k;
for(i=1;i<=n;++i) f[i]=i;
memset(d,0,sizeof(d));
while(k--){
scanf("%d%d%d",&D,&x,&y);
if(x>n||y>n||(x==y&&D==2)) {++ans;continue;}
int rx=getf(x),ry=getf(y);
if(rx==ry){
if((D-1)!=(3-d[x]+d[y])%3) ++ans;
}
else merg(x,y,rx,ry,D);
}
cout<<ans<<endl;
return 0;
}

poj 1182 (带权并查集)的更多相关文章

  1. poj 1733(带权并查集+离散化)

    题目链接:http://poj.org/problem?id=1733 思路:这题一看就想到要用并查集做了,不过一看数据这么大,感觉有点棘手,其实,我们仔细一想可以发现,我们需要记录的是出现过的节点到 ...

  2. POJ 1703 带权并查集

    直接解释输入了: 第一行cases. 然后是n和m代表有n个人,m个操作 给你两个空的集合 每个操作后面跟着俩数 D操作是说这俩数不在一个集合里. A操作问这俩数什么关系 不能确定:输出Not sur ...

  3. Navigation Nightmare POJ - 1984 带权并查集

    #include<iostream> #include<cmath> #include<algorithm> using namespace std; ; // 东 ...

  4. Parity game POJ - 1733 带权并查集

    #include<iostream> #include<algorithm> #include<cstdio> using namespace std; <& ...

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

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

  6. POJ 1182 食物链 【带权并查集】

    <题目链接> 题目大意: 动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形.A吃B, B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我 ...

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

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

  8. POJ 1182 食物链 【带权并查集/补集法】

    动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形.A吃B, B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种.有人用两种说 ...

  9. poj 1182 食物链【带权并查集】

    设相等的边权为0,吃的边权为,被吃的边权为2,然后用带权并查集在%3的意义下做加法即可 关系为简单环的基本都可以用模环长的方式是用带权并查集 #include<iostream> #inc ...

随机推荐

  1. bzoj1641 / P2888 [USACO07NOV]牛栏Cow Hurdles

    P2888 [USACO07NOV]牛栏Cow Hurdles Floyd $n<=300$?果断Floyd 给出核心式,自行体会 $d[i][j]=min(d[i][j],max(d[i][k ...

  2. phpstorm怎么设置每个function都用那条横线隔开

  3. React 回忆录(一)为什么使用 React?

    Hi 各位,欢迎来到 React 回忆录!

  4. 如何修改ls命令列出来的目录颜色

    答:默认为蓝色,在黑色背景下无法看清楚,因此以以下方法修改; 1.往~/.bash_profile文件中添加以下内容: export LS_COLORS='no=00:fi=00:di=01;33:l ...

  5. luogu3261 懒惰左偏树 [JLOI2015]城池攻占

    目录 题目 思路 错误&&反思 代码 题目 luogu 原来左偏树真的能懒惰下放 那这篇博客应该要咕咕了 一开始我按照那篇博客想了一下,感觉emm,还是瞄了一眼看到了pushdown ...

  6. EF、Repository、Factory、Service间关系

    EF和Repository 实体(Entities):具备唯一ID,能够被持久化,具备业务逻辑,对应现实世界业务对象. 值对象(Value objects):不具有唯一ID,由对象的属性描述,一般为内 ...

  7. UVa 1411 Ants(分治)

    https://vjudge.net/problem/UVA-1411 题意:n只蚂蚁和n颗苹果树,一一配对并且不能交叉. 思路:这就是巨人与鬼的问题.用分治法就行了. #include<ios ...

  8. spring boot 无法读取application.properties问题

    spring boot 无法读取application.properties问题 https://bbs.csdn.net/topics/392374488 Spring Boot 之注解@Compo ...

  9. bam/sam格式说明--转载

    在SAM输出的结果中每一行都包括十二项通过Tab分隔,从左到右分别是: 1 序列的名字(Read的名字) 2 概括出一个合适的标记,各个数字分别代表 1     序列是一对序列中的一个 2     比 ...

  10. Java I/O学习 标准的I/O重定向

    public class Test{ /* * 标准的I/O重定向 * System.setIn(InputStream) * System.setOut(PrintStream) * System. ...