A Bug's Life(种类并查集)(也是可以用dfs做)

http://acm.hdu.edu.cn/showproblem.php?pid=1829

A Bug‘s Life

Time Limit:5000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u

Submit Status Practice HDU 1829

Description

Background 
Professor Hopper is researching the sexual behavior of a rare species of bugs. He assumes that they feature two different genders and that they only interact with bugs of the opposite gender. In his experiment, individual bugs and their interactions were easy to identify, because numbers were printed on their backs.

Problem 
Given a list of bug interactions, decide whether the experiment supports his assumption of two genders with no homosexual bugs or if it contains some bug interactions that falsify it.

Input

The first line of the input contains the number of scenarios. Each scenario starts with one line giving the number of bugs (at least one, and up to 2000) and the number of interactions (up to 1000000) separated by a single space. In the following lines, each interaction is given in the form of two distinct bug numbers separated by a single space. Bugs are numbered consecutively starting from one.

Output

The output for every scenario is a line containing "Scenario #i:", where i is the number of the scenario starting at 1, followed by one line saying either "No suspicious bugs found!" if the experiment is consistent with his assumption about the bugs‘ sexual behavior, or "Suspicious bugs found!" if Professor Hopper‘s assumption is definitely wrong.

Sample Input

2
3 3
1 2
2 3
1 3
4 2
1 2
3 4

Sample Output

Scenario #1:
Suspicious bugs found!

Scenario #2:
No suspicious bugs found!

Hint

Huge input,scanf is recommended.
         

题意: 有一堆虫,正常来说只能和异性有关系,现在给你一堆关系,问你这些中可能存在“同性恋”吗

思路1: (种类并查集) 除了正常的并查集之外,此题还要维护一个关系数组记录它与其父亲的性别关系,这个关系一般自己定义,

此题可以认为是同性为 1 , 异性为0 ,那么要找到它与其父亲的父亲的关系就可以使(rank[x]+rank[x+1] )%2 ,依次递归下去就可以得到它和他最上面的祖先的关系

并查集一般就考虑并,和查两部分,其中压缩路径的时候要注意将关系数组也要更新,

 1 #include <cstdio>
 2 //存储的是其父亲的下表
 3 int bugs[2010];
 4 int relation[2010];//1:相同性别 0:不同性别
 5 //初始化
 6 void init(int len)
 7 {
 8     for(int i = 0;i <= len; i++)
 9     {
10         bugs[i] = i;
11         relation[i] = 1;
12     }
13 }
14 //找到根
15 int find(int bug)
16 {
17     if(bugs[bug]==bug)return bug;
18     int tem = bugs[bug];
19     bugs[bug] = find(bugs[bug]);//递归更新域,返回最终的父亲节点,把所有的孩子都更新了
20     //注意这里,求当前位置和父亲的关系,记录之前父亲的位置为tem,然后因为是递归,
21     //此时的relation[tem]已经在递归中更新过了,也就是孩子和父亲的关系+父亲和爷爷的关系+1然后模2就得到
22     //孩子和爷爷的关系,这里用0和1表示,0表示不同性别,1表示相同性别
23     relation[bug] = (relation[bug]+relation[tem]+1)%2;
24     return bugs[bug];
25 }
26
27 void union_set(int a,int b,int x,int y)
28 {
29     //合并,让前边的集合的根指向后边集合的根,成为一个集合
30     bugs[x]=y;
31     //更新前边集合根和新的集合根之间的关系,
32     //注意这里,relation[a]+relation[x]与relation[b]
33     //相对于新的父节点必须相差1个等级,因为他们不是gay
34     relation[x] = (relation[b]-relation[a])%2;   //这里的种类函数还是不理解,大概是根据一个关系的环状推出关系
35 }
36
37 int main()
38 {
39     int S;
40     int n,inter;
41     int bug1,bug2,parent1,parent2;
42     bool flag;//false:无同性恋,true:有同性恋
43     scanf("%d",&S);
44     for(int i=1; i<=S;i++)
45     {
46         scanf("%d%d",&n,&inter);
47         flag = false;
48         init(n);//初始化,使其父节点为自己
49         for(int j = 1; j <= inter; j++)
50         {
51             scanf("%d%d",&bug1,&bug2);
52             if(flag)continue;// 因为当有同性的时候是依次读入的数据,要保证数据时读完的所以用continue
53             parent1 = find(bug1);
54             parent2 = find(bug2);
55             if(parent1==parent2)
56             {
57                 if(relation[bug1]==relation[bug2])//同性
58                 flag = true;
59             }
60             union_set(bug1,bug2,parent1,parent2);
61         }
62         if(flag)
63         printf("Scenario #%d:\nSuspicious bugs found!\n",i);
64         else
65         printf("Scenario #%d:\nNo suspicious bugs found!\n",i);
66         printf("\n");
67     }
68     return 0;
69 }

下面是dfs的思路,将所有的关系都用链接表的形式存起来,然以后肯定会有环的情况,当时奇环的时候(定点数是基数的时候)就存在同性恋,要是偶环就是不存在同性恋,扫描的时候统计节点数,当再次扫描到同一个点的时候看扫描过得点数是奇还是偶。

 1 #include <cstdio>
 2 #include <cstring>
 3 using namespace std;
 4 #define N 2005
 5 #define M 1000005
 6
 7 int head[N];
 8 struct Edge{
 9     int v, next;
10 }edge[2*M];
11 int Ecnt;
12
13 void init()
14 {
15     Ecnt = 0;
16     memset(head, -1, sizeof(head));
17 }
18
19 void add(int u, int v)
20 {
21     edge[Ecnt].v = v;
22     edge[Ecnt].next = head[u];
23     head[u] = Ecnt++;
24     edge[Ecnt].v = u;
25     edge[Ecnt].next = head[v];
26     head[v] = Ecnt++;
27 }//存双向边
28
29 bool visited[N];
30 int f[N];//统计它是男生还是女生,定义 男生是1 女生是0 搜索的时候按0,1,0,1 的顺序标记点,如果再次访问到原先的点的时候应该按顺序标的值与其之前标的不一样的话就是奇环
31 bool dfs(int u)
32 {
33     for(int i = head[u]; i != -1; i = edge[i].next)//遍历每一条边
34     {
35         int v = edge[i].v;//下一个点
36         if(!visited[v])//如果没有访问过这个点
37         {
38             visited[v] = 1;
39             f[v] = (1-f[u]);  //按0,1 交替顺序的标记点
40             bool res = dfs(v);//如果在这之前就已经发现有奇环的话就直接返回FALSE
41             if(res == false) return false;
42         }
43         else if(f[u] == f[v]) return false;//最后又访问到了这个点,就判断这两个点是否是同一个值
44     }
45     return true;
46 }
47
48 int main()
49 {
50     int T, n, m;
51     scanf("%d", &T);
52     int cas = 1;
53     while(T--)
54     {
55         printf("Scenario #%d:\n", cas++);
56         scanf("%d %d", &n, &m);
57         init();
58         int s, t;
59         for(int i = 0; i < m; i++)
60         {
61             scanf("%d %d", &s, &t);
62             add(s, t);
63         }
64         memset(visited, 0, sizeof(visited));
65         memset(f, -1, sizeof(f));
66         bool ans = true;
67         for(int i = 1; i <= n; i++)
68         {
69             if(!visited[i])
70             {
71                 visited[i] = 1;
72                 f[i] = 1;
73                 bool res = dfs(i);
74                 if(res == false)
75                 {
76                     ans = false;
77                     break;
78                 }
79             }
80         }
81         if(ans) puts("No suspicious bugs found!\n");
82         else puts("Suspicious bugs found!\n");
83     }
84     return 0;
85 }

A Bug's Life(种类并查集)(也是可以用dfs做)

时间: 2024-11-25 13:46:51

A Bug's Life(种类并查集)(也是可以用dfs做)的相关文章

HDU 1829 &amp;&amp; POJ 2492 A Bug&#39;s Life(种类并查集)

题目地址:HDU 1829     POJ 2492 这个题可以用两种方法做,第一眼看完题是觉得用dfs染色判断二分图.然后又写的刚学的种类并查集.原来并查集可以这样用,真是神奇.. dfs染色代码: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <stdlib.h> #include <math.h> #incl

【POJ】2492 A bug&#39;s life ——种类并查集

A Bug's Life Time Limit: 10000MS   Memory Limit: 65536K Total Submissions: 28211   Accepted: 9177 Description Background Professor Hopper is researching the sexual behavior of a rare species of bugs. He assumes that they feature two different genders

hdu1829A Bug&#39;s Life(种类并查集)

传送门 关键在于到根节点的距离,如果两个点到根节点的距离相等,那么他们性别肯定就一样(因为前面如果没有特殊情况,两个点就是一男一女的).一旦遇到性别一样的,就说明找到了可疑的 1 #include<bits/stdc++.h> 2 using namespace std; 3 int f[2005],n,m,rankk[2005]; 4 bool flag; 5 inline void init() 6 { 7 flag=false; 8 for(int i=0; i<=n; ++i)

hdoj 1829 A bug&#39;s life 种类并查集

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1829 并查集的一个应用,就是检测是否存在矛盾,就是两个不该相交的集合有了交集.本题就是这样,一种虫子有两种性别,每次m次操作,每次给出(a,b),如果a和b是同性别就出现了错误,也就是说出现了判断它有两种性别的错误.我的策略同样是两个集合,用并查集维护两个集合之间的关系.具体证明请看我的上一篇博客,关于这种做法的正确性的证明. 代码如下: 1 #include<bits/stdc++.h> 2 t

poj 2492 a bug&#39;s life 简单种类并查集

题意大致为找同性恋的虫子.... 这个比食物链要简单些.思路完全一致,利用取余操作实现关系之间的递推. 个人感觉利用向量,模和投影可能可以实现具有更加复杂关系的并查集. 1 #include<cstdio> 2 using namespace std; 3 const int MAXN=50010; 4 int fa[MAXN]; 5 int rel[MAXN]; // 0代表同类,1代表吃fa[i],2代表被吃 6 void _set(int n) 7 { 8 for(int i=1;i&l

hdoj-1289-A Bug&amp;#39;s Life【种类并查集】

A Bug's Life Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 11386 Accepted Submission(s): 3709 Problem Description Background Professor Hopper is researching the sexual behavior of a rare specie

hdu1829 A Bug&#39;s Life 基础种类并查集

题目的大意可以理解为:A爱B,B爱C ……给出一系列爱恋的关系,推断有没有同性恋. 思路是把相同性别的归为一个集合,异性的异性为同性. #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int N=2010; int f[N], r[N], flag; void init() { for(int i=1;i&

hdu 1829 A Bug&#39;s Life (基础种类并查集)

先说说种类并查集吧. 种类并查集是并查集的一种.但是,种类并查集中的数据是分若干类的.具体属于哪一类,有多少类,都要视具体情况而定.当然属于哪一类,要再开一个数组来储存.所以,种类并查集一般有两个数组,一个存并查集内的父子关系,一个存各个节点所属的种类关系. 以这道题为例(题意在后面,如果没有读题,可以先看完题在来看这部分)—— 这道题很明显,将bug分成两类,一公一母.但是实际上我们并不关心它是公的还是母的,只关心它们之间是同性还是异性.所以,我们可以设与并查集的根节点同性的为0,反之为1.所

hdoj-1289-A Bug&#39;s Life【种类并查集】

A Bug's Life Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 11386 Accepted Submission(s): 3709 Problem Description Background Professor Hopper is researching the sexual behavior of a rare specie