码迷,mamicode.com
首页 > 其他好文 > 详细

食物链

时间:2017-09-05 11:19:28      阅读:158      评论:0      收藏:0      [点我收藏+]

标签:close   开始   任务   合并   有趣的   nbsp   bsp   判断   pre   

题目描述

动物王国中有三类动物 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 句话有的是真

的,有的是假的。当一句话满足下列三条之一时,这句话就是假话,否则就是真话。

? 当前的话与前面的某些真的话冲突,就是假话

? 当前的话中 X 或 Y 比 N 大,就是假话

? 当前的话表示 X 吃 X,就是假话

你的任务是根据给定的 N 和 K 句话,输出假话的总数。

输入输出格式

输入格式:

 

从 eat.in 中输入数据

第一行两个整数,N,K,表示有 N 个动物,K 句话。

第二行开始每行一句话(按照题目要求,见样例)

 

输出格式:

 

输出到 eat.out 中

一行,一个整数,表示假话的总数。

 

 

我们并不需要知道他到底是哪一种动物。。。只需要各个动物之间的相对关系即可。

那么,就是加权并查集。。。

那么有那么几种操作:

判断:

1.A与B之间关系已知,则需判断给定关系是否与之冲突。

2.未知,则按照给定关系合并。

技术分享
 1 #include<cstdio>
 2 inline int read()
 3 {
 4     int f=1,ans=0;char c=getchar();
 5     while(c<0||c>9){if(c==-)f=-1;c=getchar();}
 6     while(c>=0&&c<=9){ans=ans*10+c-0;c=getchar();}
 7     return f*ans;
 8 }
 9 const int mod=3;
10 const int MAXN=5e4+1;
11 int f[MAXN],d[MAXN],n;
12 int dis(int x,int y)
13 {
14     return (d[x]-d[y]+mod)%mod;
15 }
16 int find(int x)
17 {
18     int odf=f[x];
19     if(x!=f[x])
20     {
21         f[x]=find(f[x]);
22         d[x]=(d[x]+d[odf]+mod)%mod;
23     }//return x;
24     return f[x];
25 }
26 int jud(int z,int a,int b)
27 {
28     find(a);find(b);
29     if(a>n||b>n||(z==2&&a==b))return 1;
30     if(find(a)==find(b))
31     {
32         if(z==1&&d[a]!=d[b])return 1;
33         if(z==2&&dis(a,b)!=1)return 1;
34         return 0;
35     }
36     int dx=(z==1?0:1);
37     /*
38     f[f[b]]=f[a];
39     d[f[b]]=(d[a]-d[b]-dx+mod)%mod;
40     */
41     d[f[a]]=(mod-d[a])%mod;
42     f[f[a]]=a;
43     f[a]=b;d[a]=dx;
44     d[f[b]]=(mod-d[b])%mod;
45     f[f[b]]=b;
46     f[b]=b;d[b]=0; 
47     return 0;
48 }
49 int main()
50 {
51     n=read();
52     for(int i=1;i<=n;i++){f[i]=i;}
53     int k=read(),ans=0;
54     for(int i=1,a,b,c;i<=k;i++)
55     {
56         c=read();a=read();b=read();
57         ans+=jud(c,a,b);
58     }
59     printf("%d\n",ans);
60     return 0;
61 }
View Code

在重构f[b]与f(a)的关系时,可以用“取反”,即将a与f(a),b与f(b)的关系倒置。

这样可以避免复杂新距离的计算。

食物链

标签:close   开始   任务   合并   有趣的   nbsp   bsp   判断   pre   

原文地址:http://www.cnblogs.com/zhinv/p/7465235.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!