BZOJ 1191 超级英雄 Hero 题解

BZOJ 1191 超级英雄 Hero 题解

Description

现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金。主持人问题准备了若干道题目,只有当选手正确回答一道题后,才能进入下一题,否则就被淘汰。为了增加节目的趣味性并适当降低难度,主持人总提供给选手几个“锦囊妙计”,比如求助现场观众,或者去掉若干个错误答案(选择题)等等。 这里,我们把规则稍微改变一下。假设主持人总共有m道题,选手有n种不同的“锦囊妙计”。主持人规定,每道题都可以从两种“锦囊妙计”中选择一种,而每种“锦囊妙计”只能用一次。我们又假设一道题使用了它允许的锦囊妙计后,就一定能正确回答,顺利进入下一题。现在我来到了节目现场,可是我实在是太笨了,以至于一道题也不会做,每道题只好借助使用“锦囊妙计”来通过。如果我事先就知道了每道题能够使用哪两种“锦囊妙计”,那么你能告诉我怎样选择才能通过最多的题数吗?

Input

输入文件的一行是两个正整数n和m(0 < n <1001,0 < m < 1001)表示总共有n中“锦囊妙计”,编号为0~n-1,总共有m个问题。
以下的m行,每行两个数,分别表示第m个问题可以使用的“锦囊妙计”的编号。
注意,每种编号的“锦囊妙计”只能使用一次,同一个问题的两个“锦囊妙计”可能一样。

Output

第一行为最多能通过的题数p

Sample Input

5 6
3 2
2 0
0 3
0 4
3 2
3 2

Sample Output

4

————————————————————————分割线————————————————————————

这道是一道二分图匹配裸题,匈牙利算法模板题。

建议使用邻接表储存边,时间复杂度较低。

代码如下:

 1 /**************************************************************
 2     Problem: 1191
 3     User: shadowland
 4     Language: C++
 5     Result: Accepted
 6     Time:40 ms
 7     Memory:1692 kb
 8 ****************************************************************/
 9
10 #include "bits/stdc++.h"
11
12 using namespace std ;
13 const int maxN = 10100 ;
14 struct Match {int to , next ;};
15
16 Match E[ maxN<<2 ] ;
17 int head [ maxN ] , match[ maxN ] ;
18 bool vis [ maxN ] ;
19
20 int cnt = 0 ,ans = 0 ;
21
22 void Add_Edge ( int x , int y ) {
23          E[ ++cnt ] . to = y ;
24          E[ cnt ] . next = head[ x ] ;
25          head [ x ] = cnt ;
26 }
27
28 bool Hungary ( int x ) {
29          for ( int i = head[ x ] ; i ; i = E[ i ] . next ) {
30                  if ( vis[ i ] ) continue ;
31                  int temp = E[ i ] . to ;
32                  vis[ i ] = true ;
33                  if ( !match[ temp ] || Hungary ( match[ temp ] ) ) {
34                           match [ temp ] = x ;
35                           return true ;
36                  }
37          }
38          return false ;
39 }
40
41 int main ( ) {
42          int N , M ;
43          memset ( match , 0 , sizeof ( match ) ) ;
44          scanf ( "%d %d" , &N , &M ) ;
45          for ( int i=1 ; i<=M ; ++i ) {
46                  int x1 , x2 ;
47                  scanf ( "%d%d" , &x1 , &x2 ) ;
48                  Add_Edge ( i , x1 ) , Add_Edge ( i , x2 ) ;//使用邻接表储存
49          }
50          int k ;
51          for ( k=1 ; k<=M ; ++k ) {
52                  memset ( vis , false , sizeof ( vis ) ) ;
53                  if ( !Hungary ( k ) )
54                          break ;
55          }
56          printf ( "%d\n" , k - 1 ) ;
57          return 0 ;
58 }

2016-09-16 16:00:59

(完)

时间: 2024-10-05 11:04:54

BZOJ 1191 超级英雄 Hero 题解的相关文章

bzoj 1191 超级英雄Hero

题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1191 题解: 裸匈牙利,注意如果出现找不到增广路的情况就直接break 1 #include<cstdio> 2 #include<cstring> 3 #define MAXN 1010 4 int n,m,cnt,match[MAXN*2],head[MAXN*2],ans; 5 bool check[MAXN*2]; 6 struct node 7 { 8 int v

BZOJ 1191 超级英雄(二分图匹配)

把题目作为s集,锦囊作为t集.把每个题目和它可以用的锦囊连边,这样就构成了一个二分图,求出这个二分图最大匹配. 但是这个最大匹配有限制条件,就是对于每个可能的匹配集,如果s集的i点有匹配,那么i-1点一定有匹配. 具体实现,只需要将匈牙利算法稍微改动一下,如果当前没有找到增广路的话,就break. # include <cstdio> # include <cstring> # include <cstdlib> # include <iostream> #

BZOJ 1191 超级英雄

同SCOI2010 游戏. #include<iostream> #include<cstdio> #include<cstring> #define maxv 2050 #define maxe 200050 using namespace std; struct edge { int v,nxt; }e[maxe]; int n,m,a,b,g[maxv],nume=0,vis[maxv],linker[maxv]; void addedge(int u,int v

BZOJ 1191: [HNOI2006]超级英雄Hero 二分匹配

1191: [HNOI2006]超级英雄Hero Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目,只有当选手正确回答一道题后,才能进入下一题,否则就被淘汰.为了增加节目的趣味性并适当降低难度,主持人总提供给选手几个“锦囊妙计”,比如求助现场观众,或者去掉若干个错误答案(选择题)等等. 这里,我们把规则稍微改变一下.假设主持人总共有m道题,选手有n种不同的“锦囊妙

bzoj 1191: [HNOI2006]超级英雄Hero 并查集 || 匈牙利算法

1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1804  Solved: 850[Submit][Status] Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目,只有当选手正确回答一道题后,才能进入下一题,否则就被淘汰.为了增加节目的趣味性并适当降低难度,主持人总

BZOJ 1191: [HNOI2006]超级英雄Hero(二分图匹配)

云神说他二分图匹配从来都是用网络流水过去的...我要发扬他的精神.. 这道题明显是二分图匹配.网络流的话可以二分答案+最大流.虽然跑得很慢.... ---------------------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> #include<iostrea

1191: [HNOI2006]超级英雄Hero

1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1996  Solved: 946[Submit][Status] Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目,只有当选手正确回答一道题后,才能进入下一题,否则就被淘汰.为了增加节目的趣味性并适当降低难度,主持人总

1191: [HNOI2006]超级英雄Hero 二分图匹配(?)

1191: [HNOI2006]超级英雄Hero Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目,只有当选手正确回答一道题后,才能进入下一题,否则就被淘汰.为了增加节目的趣味性并适当降低难度,主持人总提供给选手几个"锦囊妙计",比如求助现场观众,或者去掉若干个错误答案(选择题)等等. 这里,我们把规则稍微改变一下.假设主持人总共有m道题,选手有n种不同

矩阵乘法专题1——bzoj 1297 [SCOI2009] 迷路题解

题目链接 题意:给两个长度分别为n和m的序列,现在有两种操作:1.分别选择两个序列的一个非空前缀,切两个前缀的最后一位相同,删除之,得到1分(只累计),消耗e:2.直接删除两个序列,消耗值定于两个序列之前删除的元素个数之和,并且使得得到的分有效(之前没有有效分) 分析: 首先,问题其实就是转化成,进行若干次操作1,然后进行操作2 还要找到一个判别标准,来评判较优的状态(贪心) 每次的消耗值比较大,其实可以计算出最大的删除次数,这个值不是很大 状态表示: 简单的,一个状态可以表示为串A的位置.串B