食物链
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 60225   Accepted: 17656

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
 
思路:
根据题意,可以肯定的是一个一题关于集合的题目,所以考虑并查集处理。由于这里并不是直接的种类,然后动物。所以给可以给当前的动物和他的父亲之间一个值来表示关系。
现在pa[i]表示i的,rel[i]表示pa[i] 和 i的关系。rel[i] = 0表示为同一类,1表示i可以吃i吃pa[i],2表示pa[i] 吃 i。
对于输入的x,y。先找到他们所在集合的祖先,也就是pa[i](因为路径压缩了,所以pa[i]不是表示i的父亲)。fx=pa[x],fy=pa[y]。
     1.如果fx == fy说明这x,y之前已经处理过,已经在同一个集合之中。只要判断他们2个在集合中的关系就好。
 
   如图所示,现在x=3,y=4,x y已经在同一个集合中,现在要根据已经知道的rel[3],rel[4],来得到3和4的关系。不过先要知道3 - rel[4]就是图中1到4的箭头,由于传递性,3到4的关系就是(rel[3] + 3 - rel[4]) % 3,这样就可以得到3 4之间的关系,根据这个关系来判断是否这句话正确。 
     2.如果x,y不在同一个集合中。说明这句话是对的,同时我们需要对值进行更新维护。我在这里更新都是pa[fy] = fx;
如图所以现在x = 2,y = 5,x,y不在同一个集合中。这时候我把pa[fy] = fx,这样的话我要更新的是rel[fy],此时rel[fy] = (3 - rel[y] + j + rel[x]) % 3,这里的j是根据d来的,如果d=1,那么j = 0,因为同类,如果d = 2,那么j = 2,因为x 能够吃 y,所以y 和 x的关系为2。
这样就完成了2个集合的合并。
 
现在还有一部就是路径压缩时候也要更新,维护rel的值。很简单,因为递归,先更新了pa[x],这时候rel[pa[x]]更新了,rel[pa[x]]为pa[x]指向集合根节点的rel值了。所以rel[x] = (rel[x] + rel[pa[x]]) % 3。这样问题就解决了。
 
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<string>
#include<time.h>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define INF 1000000001
#define ll long long
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
using namespace std;
const int MAXN = ;
int pa[MAXN],n,k,rel[MAXN];
void Init()
{
for(int i = ; i <= n; i++){
pa[i] = i;
}
memset(rel,,sizeof(rel));
}
int find(int x)
{
if(x != pa[x]){
int fx = find(pa[x]);
rel[x] = (rel[x] + rel[pa[x]]) % ;
pa[x] = fx;
}
return pa[x];
}
int main()
{
scanf("%d%d",&n,&k);
Init();
int d,x,y;
int ans = ;
while(k--){
scanf("%d%d%d",&d,&x,&y);
if(x > n || y > n){
ans ++;
}
else if(d == && x == y){
ans ++;
}
else if(d == ){
int fx = find(x);
int fy = find(y);
if(fx == fy){
int ret = (rel[x] + - rel[y]) % ;
if(ret != ){
ans ++;
}
}
else {
pa[fy] = fx;
rel[fy] = ( - rel[y] + rel[x]) % ;
}
}
else {
int fx = find(x);
int fy = find(y);
if(fx == fy){
int ret = (rel[x] + - rel[y]) % ;
if(ret != ){
ans ++;
}
}
else {
pa[fy] = fx;
rel[fy] = ( - rel[y] + rel[x]) % ;
rel[fy] = rel[fy];
}
}
//cout<<ans<<endl;
}
printf("%d\n",ans);
return ;
}

poj1182 带权并查集的更多相关文章

  1. 种类并查集——带权并查集——POJ1182;HDU3038

    POJ1182 HDU3038 这两个题比较像(一类题目),属于带权(种类)并查集 poj1182描绘得三种动物种类的关系,按照他一开始给你的关系,优化你的种类关系网络,最后看看再优化的过程中有几处矛 ...

  2. poj1182 食物链(带权并查集)

    题目链接 http://poj.org/problem?id=1182 思路 前面做的带权并查集的权值记录该结点与其父结点是否是同一类,只有两种取值情况(0,1),在这题中某结点a和其父结点b的取值共 ...

  3. 【带权并查集】poj1182 食物链

    带权并查集,或者叫做种类并查集,经典题. http://blog.csdn.net/shuangde800/article/details/7974668 这份代码感觉是坠吼的. 我的代码是暴力分类讨 ...

  4. 【POJ1182】 食物链 (带权并查集)

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

  5. 带权并查集 poj1182

    首先要注意核心代码 int find(int i){    if(i == fa[i])        return fa[i];    int tt = find(fa[i]);    num[i] ...

  6. poj1182食物链,经典带权并查集

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

  7. poj1182(带权并查集)

    题目链接:http://poj.org/problem?id=1182 题意:题目告诉有  3  种动物,互相吃与被吃,现在告诉你  m  句话,其中有真有假,叫你判断假的个数  (  如果前面没有与 ...

  8. Poj1182 食物链(并查集/带权并查集)

    题面 Poj 题解 这里采用并查集的补集. \(x\)表示同类集合,\(x+n\)表示敌人集合,\(x+n\times2\)表示敌人的敌人集合. 如果当前给出的是一对同类关系,就判断\(x\)是否吃\ ...

  9. poj1182 食物链 带权并查集

    题目传送门 题目大意:大家都懂. 思路: 今天给实验室的学弟学妹们讲的带权并查集,本来不想细讲的,但是被学弟学妹们的态度感动了,所以写了一下这个博客,思想在今天白天已经讲过了,所以直接上代码. 首先, ...

随机推荐

  1. sql 盲注之正则表达式攻击

    -----------------------------------------MYSQL 5+----------------------------------------- 我们都已经知道,在 ...

  2. webpack中output配置项中chunkFilename属性的用法

    chunkFilename和webpack.optimize.CommonsChunkPlugin插件的作用差不多,都是用来将公共模块提取出来,但是用法不一样,这里主要介绍chunkFilename的 ...

  3. Orchard CRM 更新 - 同时支持 Microsoft Dynamics CRM 2011, 2013, 2015, 2016!

    本版本支持: 使用Orchard 1.8.1 系统 Dynamics CRM 2015 DLL .Net Framework 4.5.2 演示版本: http://www.orchardcrm.com ...

  4. 两步验证Authy时间同步问题

    Authy是我常用的软件之一,通常用于Google的两步验证,或者是其他基于Google两步验证的原理的衍生程序.比如Namesilo.印象笔记等均有使用. 先说说什么是两步验证. 两步验证 两步验证 ...

  5. DOM 概况

    DOM(文档对象模型)是针对 HTML 和 XML 文档的一个API(应用程序编程接口).DOM 描绘了一个层次化的节点树,允许开发人员添加.移除和修改页面的某一部分. 层次节点 DOM可以将任何 H ...

  6. 关于mvc5+EF里面的db.Entry(model).State = EntityState.Modified报错问题

    最近在使用mvc5+EF的的时候用到了这句话 db.Entry(model).State = EntityState.Modified 看上去很简单的修改数据,但是一直报错,说是key已经存在,不能修 ...

  7. unity3d 音频无缝循环

    在我做赛车漂移的时候,漂移的声音断断续续的,搞得我很郁闷 大家可以随便找个音效然后循环播放去仔细听 你会发现当音效播放完成一次之后循环播放第二次时会停顿一下 我做赛车漂移如果中途停顿了体验是非常不好的 ...

  8. ArcEngine选中面要素样式修改

    //只用前五行,可以直接将选中的面要素的颜色全部修改成红色,也就是填充颜色 IRgbColor pRgbColor= new RgbColor();; pRgbColor.Red = ; pRgbCo ...

  9. Code Review 五问五答

    Code Review 是什么? Code Review即代码审查,程序猿相互审核对方的代码. Code Review能获得什么好处? 提高代码可维护性 你写的代码不再只有编译器看了,你得写出审核人能 ...

  10. CSS高级知识

    1.CSS变换 2.CSS动画 3.CSS高级特性及兼容性:http://caniuse.com/