codevs 1222 信与信封问题(二分图的完美匹配)

1222 信与信封问题

题目描述 Description

John先生晚上写了n封信,并相应地写了n个信封将信装好,准备寄出。但是,第二天John的儿子Small John将这n封信都拿出了信封。不幸的是,Small John无法将拿出的信正确地装回信封中了。

将Small John所提供的n封信依次编号为1,2,…,n;且n个信封也依次编号为1,2,…,n。假定Small John能提供一组信息:第i封信肯定不是装在信封j中。请编程帮助Small John,尽可能多地将信正确地装回信封。

输入描述 Input Description

n文件的第一行是一个整数n(n≤100)。信和信封依次编号为1,2,…,n。

n接下来的各行中每行有2个数i和j,表示第i封信肯定不是装在第j个信封中。文件最后一行是2个0,表示结束。

输出描述 Output Description

输出文件的各行中每行有2个数i和j,表示第i封信肯定是装在第j个信封中。请按信的编号i从小到大顺序输出。若不能确定正确装入信封的任何信件,则输出“none”。

样例输入 Sample Input

3

1  2

1  3

2  1

0  0

样例输出 Sample Output

1   1

【思路】

二分图的完美匹配。

先一遍match看是否有能够完美匹配,没有则none。如果有枚举删除完美匹配中的边,如果删边后该点不能匹配则输出该边,如果没有则none。


 1 #include<cstdio>
 2 #include<cstring>
 3 #include<queue>
 4 #include<vector>
 5 #define FOR(a,b,c) for(int a=(b);a<(c);a++)
 6 using namespace std;
 7
 8 const int maxn = 100+10;
 9
10 bool T[maxn];
11 int lkx[maxn],lky[maxn];
12 //lkx 表示与X结点匹配的Y结点 lky表示与Y结点相匹配的X结点
13 int n,m;
14 int A[maxn][maxn];
15
16 bool match(int u) {
17     for(int v=1;v<=n;v++)
18         if(!A[u][v] && !T[v]) {
19             T[v]=1;
20             if(!lky[v] || match(lky[v])) {
21                 lky[v]=u , lkx[u]=v;
22                 return true;
23             }
24         }
25     return false;
26 }
27
28 int main() {
29     scanf("%d",&n);
30     int u,v;
31     while(scanf("%d%d",&u,&v)==2 && (u&&v))
32         A[u][v]=1;
33     int ans=0;
34     for(int i=1;i<=n;i++) {
35         memset(T,0,sizeof(T));
36         if(match(i)) ans++;
37     }
38     if(ans!=n) printf("none\n");
39     else {
40         bool flag=0;
41         for(int i=1;i<=n;i++) {
42             int v=lkx[i];
43             A[i][v]=1;
44             lkx[i]=0 , lky[v]=0;
45             memset(T,0,sizeof(T));
46             if(!match(i)) {
47                 printf("%d %d\n",i,v);
48                 lkx[i]=v,lky[v]=i; flag=1;    //恢复
49             }
50             A[i][v]=0;                        //恢复
51         }
52         if(!flag) puts("none\n");
53     }
54     return 0;
55 }
时间: 2024-10-11 06:14:03

codevs 1222 信与信封问题(二分图的完美匹配)的相关文章

codevs 1222 信与信封问题

/* 二分图 题目给出的是确定不连通的边 如果我们剩下的可能联通也可能不连通的 */ #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #define maxn 210 using namespace std; int n,num,head[maxn],g[maxn][maxn],ans,f[maxn],match[maxn],sum,ei; struct An

WIKIOI 1222信与信封问题

题目描述 Description John先生晚上写了n封信,并相应地写了n个信封将信装好,准备寄出.但是,第二天John的儿子Small John将这n封信都拿出了信封.不幸的是,Small John无法将拿出的信正确地装回信封中了. 将Small John所提供的n封信依次编号为1,2,…,n:且n个信封也依次编号为1,2,…,n.假定Small John能提供一组信息:第i封信肯定不是装在信封j中.请编程帮助Small John,尽可能多地将信正确地装回信封. 输入描述 Input Des

HDU_2255 二分图最佳完美匹配 KM匈牙利算法

一开始还没看懂这个算法,后来看了陶叔去年的PPT的实例演示才弄懂 用一个lx[]和ly[]来记录X和Y集合中点的权值,有个定理是 lx[i]+ly[j]==w[i][j](边权值) 则该点是最佳匹配,因为首先 那个不等式肯定要>=的,否则就不满足题意了,如果是>则可以去匹配更有价值的边或者把权值降下来让匹配边的潜力更大. 所以只有把握了这个条件,其他就是走一遍最大匹配数.以及up()函数用来在无法匹配的时候,进行其他点的权值降低(也可以说是增广路的搜索)来得到匹配. #include <

UVa 11383 少林决胜(二分图最佳完美匹配)

https://vjudge.net/problem/UVA-11383 题意: 给定一个N×N矩阵,每个格子里都有一个正整数W(i,j).你的任务是给每行确定一个整数row(i),每列也确定一个整数col(i),使得对于任意格子(i,j),w(i,j)<=row(i)+col(j).所有的row(i)和col(i)只和应尽量小. 思路: 利用二分图最佳完美匹配当中的l(x)+l(y)>=w(i,j),直接用KM算法即可. 1 #include<iostream> 2 #inclu

UVA - 1045 The Great Wall Game(二分图最佳完美匹配)

题目大意:给出棋盘上的N个点的位置.如今问将这些点排成一行或者一列.或者对角线的最小移动步数(每一个点都仅仅能上下左右移动.一次移动一个) 解题思路:暴力+二分图最佳完美匹配 #include <cstdio> #include <cstring> #define N 20 #define INF 0x3f3f3f3f #define abs(x) ((x) > 0 ? (x) : (-(x))) #define max(a,b)((a)>(b)? (a):(b)) #

HDU 3081--【二分图 &amp;&amp; 传递闭包 &amp;&amp; 完美匹配次数 &amp;&amp; 经典】

Marriage Match II Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 2794    Accepted Submission(s): 938 Problem Description Presumably, you all have known the question of stable marriage match. A

UVALive 4043 Ants 蚂蚁(二分图最佳完美匹配)

题意:n个蚂蚁n棵树,蚂蚁与树要配对,在配对成功的一对之间连一条线段,要求所有线段不能相交.按顺序输出蚂蚁所匹配的树. 思路:这个题目真是技巧啊,如果知道要求的就是整体最优,那么就容易做了.而不能用贪心来为每个蚂蚁选择最近的树,这样可能八成还是相交了. 整体最优能让每条线段不相交,证明: 假设a1-b1与a2-b2相交.则dis(a1,b1)+dis(a2,b2)>=dis(a1,b2)+dis(a2,b1).如果我们所决定的最优匹配是按照整体距离最短来匹配的,那么dis(a1,b1)+dis(

HDU 3722 Card Game(二分图最佳完美匹配+KM算法)

题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3722 1 /* 2 问题 3 将任意的两个字符串进行匹配,使得匹配后权值和最大 4 5 解题思路 6 将任意的字符串的权值计算出来,使用KM算法即可. 7 */ 8 #include<cstdio> 9 #include<cstring> 10 #include<algorithm> 11 using namespace std; 12 13 const int maxn

codevs1222 信与信封问题

1222 信与信封问题 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description John先生晚上写了n封信,并相应地写了n个信封将信装好,准备寄出.但是,第二天John的儿子Small John将这n封信都拿出了信封.不幸的是,Small John无法将拿出的信正确地装回信封中了. 将Small John所提供的n封信依次编号为1,2,…,n:且n个信封也依次编号为1,2,…,n.假定Small John能提供一组信息:第i封信肯