HDU 5285 wyh2000 and pupil (二分图)

题意:共有n个小学生,编号为1−n。将所有小学生分成2组,每组都至少有1个人。但是有些小学生之间并不认识,而且如果a不认识b,那么b也不认识a。Wyh2000希望每组中的小学生都互相认识。而且第一组的人要尽可能多。请你帮wyh2000求出第一组和第二组的人数是多少。如果找不到分组方案,则输出"Poor wyh"。

思路:二分图着色。给的就是无向图,每次都累加人多的颜色即可。若不能着色,必定不能分成2组。如果全部都是1个颜色,那么要让其中1人过第2组。我勒个去,就因为赭色时颜色号码开小了而错。

  1 //#include <bits/stdc++.h>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <iostream>
  5 #include <map>
  6 #include <set>
  7 #include <deque>
  8 #include <algorithm>
  9 #include <vector>
 10 #define LL long long
 11 #define pii pair<int,int>
 12 #define INF 0x7f7f7f7f
 13 using namespace std;
 14 const int N=100000+100;
 15
 16 vector<int> vect[N];
 17 int col[N];
 18 set<int> mapp;
 19
 20 int color(int u, int f)
 21 {
 22     mapp.clear();
 23     deque<int> que;
 24     que.push_back(u);
 25     col[u]=f;
 26     mapp.insert(u);
 27     while(!que.empty())
 28     {
 29         int x=que.front();
 30         que.pop_front();
 31
 32         for(int i=0; i<vect[x].size(); i++)
 33         {
 34             int tmp=vect[x][i];
 35             if(col[tmp]==col[x]) return 0;  //冲突
 36             if(col[tmp]==0)   //没色
 37             {
 38                 mapp.insert(tmp);//一个连同分量装进去
 39                 col[tmp]=1000000-col[x];
 40                 que.push_back(tmp);
 41             }
 42         }
 43     }
 44     return 1;
 45 }
 46
 47
 48 int cal(int n)
 49 {
 50     memset(col,0,sizeof(col));
 51     int big=0, small=0, k=0;
 52     for(int i=1; i<=n; i++)
 53     {
 54         if(!col[i])
 55         {
 56             if(!color(i, ++k)) return 0;
 57
 58             //统计人数
 59             int a=0, b=0;
 60             set<int>::iterator it=mapp.begin();
 61             for(int j=0; j<mapp.size(); j++)
 62             {
 63                 if(col[*it]==k)   a++;
 64                 if(col[*it]==1000000-k) b++;
 65                 it++;
 66             }
 67
 68             if(a<b) swap(a, b);
 69             big+=a;
 70             small+=b;
 71         }
 72     }
 73     if(!small)  return big-1;
 74     else return big;
 75 }
 76
 77
 78
 79 int main()
 80 {
 81     freopen("input.txt", "r", stdin);
 82     int t, n, m, a, b;
 83     cin>>t;
 84     while(t--)
 85     {
 86         scanf("%d%d",&n,&m);
 87         for(int i=0; i<=n; i++)    vect[i].clear();
 88
 89         for(int i=0; i<m; i++)
 90         {
 91             scanf("%d%d",&a,&b);
 92             vect[a].push_back(b);
 93             vect[b].push_back(a);
 94         }
 95         if(n<=1)
 96         {
 97              puts("Poor wyh");
 98              continue;
 99         }
100         else if(n==2)
101         {
102              puts("1 1");
103              continue;
104         }
105
106         int tmp=cal(n);
107         if(tmp) printf("%d %d\n",tmp, n-tmp);
108         else  puts("Poor wyh");
109
110
111     }
112     return 0;
113 }

AC代码

时间: 2024-11-07 10:51:32

HDU 5285 wyh2000 and pupil (二分图)的相关文章

HDU 5285 wyh2000 and pupil (DFS染色判二分图 + 贪心)

wyh2000 and pupil Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others) Total Submission(s): 1040    Accepted Submission(s): 331 Problem Description Young theoretical computer scientist wyh2000 is teaching his pupils. W

HDU 5285 wyh2000 and pupil(dfs或种类并查集)

wyh2000 and pupil Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others) Total Submission(s): 755    Accepted Submission(s): 251 Problem Description Young theoretical computer scientist wyh2000 is teaching his pupils. Wy

Hdu 5285 wyh2000 and pupil (bfs染色判断奇环) (二分图匹配)

题目链接: BestCoder Round #48 ($) 1002 题目描述: n个小朋友要被分成两班,但是有些小朋友之间是不认得的,所以规定不能把不认识的小朋友分在一个班级里面,并且一班的人数要比二班的人数多,每个班的人数都大于零. 解题思路: hdu给出的题解是二分图匹配加上贪心,就不多说了. 还可以用bfs对节点染色,建好图后,对节点进行bfs分成,偶数成与奇数成染成不同的颜色,颜色相同的节点都可以分到同一个集合里面,但是要判断一下奇环,如果出现奇环的话,是无法进行分组的.在每次bfs的

hdu 5285 wyh2000 and pupil(二分图判定)

对每两个不认识的人连一条边,则此题可转化为二分图判定(二分图可有多个). 如果有一部分图判定为不是二分图,则输出“Poor wyh”. 否则,分别累加每个二分图的最多的颜色数. #include <algorithm> #include <iostream> #include <cstring> #include <cstdio> #include <string> #include <stack> #include <cmat

HDU 5285 wyh2000 and pupil

题意:有一群人,已知某两人之间互相不认识,要把这群人分成两部分,每部分至少一人,且在每部分内没有人互不认识. 解法:图染色.某场bestcoder第二题……看完题觉得是个二分图……完全不会二分图什么的……但是为了挣扎一下百度了一下二分图的判定方法,知道了可以用染色法,这样如果是二分图的话将每个连通分量里点数量最多的颜色的点数量(像个绕口令诶)相加就可以了.然而激动万分的我早忘了还有每部分至少一人这个条件……直到我和队友研究怎么hack别人的时候他才告诉我还有这么个条件……(哭)还好来得及…… 代

hdu 5285 wyh2000 and pupil(二染色)

第一次用vector解得题,值得纪念,这道题是二染色问题,我用bfs解得,就是染色,判断,计数问题,其 实挺简单的,就是得判一下特殊情况,当n<2的时候就不能有解,因为题目要求每个组至少有一个人,当没有不认识的 人的时候就是一个组是n-1,另一个组人数为1 上代码: #include<stdio.h> #include<string.h> #include<stdlib.h> #include<algorithm> #include<vector

hdu 5258 wyh2000 and pupil(dfs)(待续)

题意:n个点,m条边,每条边连接的两点颜色不同, 思路: #include<cstdio> #include<cstring> #include<algorithm> #include<cmath> #include<vector> using namespace std; int t,n,m; vector<int>g[500010]; int vis[500010]; int cnt1,cnt2,sum; int dfs(int

二分图判定+点染色 BestCoder Round #48 ($) 1002 wyh2000 and pupil

题目传送门 1 /* 2 二分图判定+点染色:因为有很多联通块,要对所有点二分图匹配,若不能,存在点是无法分配的,no 3 每一次二分图匹配时,将点多的集合加大最后第一个集合去 4 注意:n <= 1,no,两个集合都至少有一人:ans == n,扔一个给另一个集合 5 */ 6 #include <cstdio> 7 #include <algorithm> 8 #include <cstring> 9 #include <cmath> 10 #in

wyh2000 and pupil

wyh2000 and pupil Accepts: 93 Submissions: 925 Time Limit: 3000/1500 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others) 问题描写叙述 青年理论计算机科学家wyh2000在教导他的小学生. 共同拥有个小学生,编号为.为了添加小学生之间的凝聚力,wyh2000决定将全部小学生分成组,每组都至少有个人. 可是有些小学生之间并不认识,并且假设不认识,那么也不认识. W