题目背景

第二次世界大战时期..

题目描述

英国皇家空军从沦陷国征募了大量外籍飞行员。由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其中1 名是英国飞行员,另1名是外籍飞行员。在众多的飞行员中,每一名外籍飞行员都可以与其他若干名英国飞行员很好地配合。如何选择配对飞行的飞行员才能使一次派出最多的飞机。对于给定的外籍飞行员与英国飞行员的配合情况,试设计一个算法找出最佳飞行员配对方案,使皇家空军一次能派出最多的飞机。

对于给定的外籍飞行员与英国飞行员的配合情况,编程找出一个最佳飞行员配对方案,使皇家空军一次能派出最多的飞机。

输入输出格式

输入格式:

第 1 行有 2 个正整数 m 和 n。n 是皇家空军的飞行员总数(n<100);m 是外籍飞行员数(m<=n)。外籍飞行员编号为 1~m;英国飞行员编号为 m+1~n。

接下来每行有 2 个正整数 i 和 j,表示外籍飞行员 i 可以和英国飞行员 j 配合。最后以 2个-1 结束。

输出格式:

第 1 行是最佳飞行员配对方案一次能派出的最多的飞机数 M。接下来 M 行是最佳飞行员配对方案。每行有 2个正整数 i 和
j,表示在最佳飞行员配对方案中,飞行员 i 和飞行员 j 配对。如果所求的最佳飞行员配对方案不存在,则输出‘No Solution!’。

输入输出样例

输入样例#1:

5 10
1 7
1 8
2 6
2 9
2 10
3 7
3 8
4 7
4 8
5 10
-1 -1

输出样例#1:

4
1 7
2 9
3 8
5 10

Solution:

本题有两种方法。

方法一:明眼看,这道题一道裸的二分图最大匹配的题目,只需要判断并输出前趋数组pre和对应的i就OK了。

代码:

 #include<bits/stdc++.h>
using namespace std;
const int N=;
inline int gi()
{
char x=getchar();int a=,f=;
while(x<''||x>''){if(x=='-')f=-;x=getchar();}
while(x>=''&&x<=''){a=a*+x-'';x=getchar();}
return a*f;
}
int n,m,s,u,v;
struct edge
{
int v,ne;
}e[N*N<<];
int h[N],cnt;
inline void ins(int u,int v){
e[++cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;
}
int le[N];
bool vis[N];
bool find(int u){
for(int i=h[u];i;i=e[i].ne){
int v=e[i].v;
if(!vis[v]){
vis[v]=;
if(!le[v]||find(le[v])){
le[v]=u;
return ;
}
}
}
return ;
}
int ans;
int main(){
n=gi();m=gi();
while(){u=gi();v=gi();if(v==-)break;ins(u,v);}
for(int i=;i<=m;i++){
memset(vis,,sizeof(vis));
if(find(i)) ans++;
}
if(ans==)printf("No Solution!");
else {
printf("%d\n",ans);
for(int i=n+;i<=m;i++)
if(le[i])printf("%d %d\n",le[i],i);
}
return ;
}

方法二:学了网络流后,这道题也能用最大流做了。我们只需建立两个虚点S和T,S连向其中一个点集建n条边,T连向另一个点集建m条边,上述的边设置容量为1,然后照常对两个点集进行建边就行了(边设置容量为1),最后跑最大流就行了,当然也得记录并输出前趋。

代码(1):(非手打,搬运自巨佬YZK的ISAP)

 #include<bits/stdc++.h>
#define sins(x,y,z) ins(x,y,z),ins(y,x,0)
#define INF (0x7fffffff)
#define DEBUG printf("Passing %d in [%s]...\n",__LINE__,__FUNCTION__)
#define _rg register
using namespace std; template <typename T> void gn(T &x){
x=;
T pl=;
char ch=getchar();
while(ch<'' || ch>'') pl=(ch=='-'?-:), ch=getchar();
while(ch>='' && ch<='') x=x*+ch-'', ch=getchar();
x*=pl;
} const int N=,M=2e5+;
int h[N],to[M],nexp[M],W[M],p=;
inline void ins(int a,int b,int w){
nexp[p]=h[a], h[a]=p, to[p]=b, W[p]=w, p++;
} int m,n;
int fl[N],rem[N];
int ISAP(int x,int op){
if(x==n+) return op;
int flow=,d,minfl=-;
// DEBUG;
for(_rg int u=h[x];u;u=nexp[u])
if(W[u]){
if(fl[to[u]]+==fl[x]) {
d=ISAP(to[u], op-flow<W[u]?op-flow:W[u]);
flow+=d, W[u]-=d, W[u^]+=d;
if(flow==op || fl[n+]==-) return flow;
}
if((minfl==- || minfl>fl[to[u]]+) && fl[to[u]]!=-) minfl=fl[to[u]]+;
}
// DEBUG;
if(!flow){
if(!(--rem[fl[x]])) fl[n+]=-;
fl[x]=minfl;
if(fl[x]!=-) rem[fl[x]]++;
}
return flow;
} int main(){
// ios::sync_with_stdio(false);
gn(m),gn(n);
int u,v;
while(gn(u),gn(v),u!=-) sins(u,v,);
for(int i=;i<=m;i++) sins(n+,i,);
for(int i=m+;i<=n;i++) sins(i,n+,); queue<int> qu;
qu.push(n+);
fl[n+]=, rem[]++;
while(!qu.empty()){
int na=qu.front();
qu.pop();
for(int u=h[na];u;u=nexp[u])
if(!W[u] && !fl[to[u]]){
fl[to[u]]=fl[na]+;
rem[fl[na]+]++;
qu.push(to[u]);
}
}
// DEBUG;
int ans=;
if(!fl[n+]) cout<<"No Solution!"<<endl;
while(fl[n+]!=-) ans+=ISAP(n+,INF);
cout<<ans<<endl;
for(int i=;i<=m;i++){
for(int u=h[i];u;u=nexp[u])
if(to[u]!=n+ && !W[u]) {cout<<i<<' '<<to[u]<<endl; break;}
}
return ;
}

代码2:(纯手打Dinic)

 #include<bits/stdc++.h>
#define il inline
using namespace std;
const int N=,inf=;
int m,n,dis[N],h[N],cnt=,ans,s,t;
struct edge{
int to,net,cap;
}e[];
il void add(int u,int v,int w)
{
e[++cnt].to=v,e[cnt].net=h[u],h[u]=cnt,e[cnt].cap=w;
e[++cnt].to=u,e[cnt].net=h[v],h[v]=cnt,e[cnt].cap=;
}
queue<int>q;
il bool bfs()
{
memset(dis,-,sizeof(dis));
dis[s]=;
q.push(s);
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=h[u];i;i=e[i].net)
{
int v=e[i].to;
if(dis[v]==-&&e[i].cap>){dis[v]=dis[u]+;q.push(v);}
}
}
return dis[t]!=-;
}
il int dfs(int u,int op)
{
if(u==t)return op;
int flow=,tmp=;
for(int i=h[u];i;i=e[i].net)
{
int v=e[i].to;
if(dis[v]==dis[u]+&&e[i].cap>){
tmp=dfs(v,min(op,e[i].cap));
if(!tmp)continue;
op-=tmp;flow+=tmp;
e[i].cap-=tmp;e[i^].cap+=tmp;
if(!op)break;
}
}
if(!flow)dis[u]=-;
return flow;
}
int main()
{
scanf("%d%d",&m,&n);
s=,t=n+;
int u,v,w;
while(){
scanf("%d%d",&u,&v);
if(u==-&&v==-)break;
add(u,v,);
}
for(int i=;i<=m;i++)add(s,i,);
for(int i=m+;i<=n;i++)add(i,n+,);
while(bfs())ans+=dfs(s,inf);
if(!ans)printf("No Solution!");
else {
printf("%d\n",ans);
for(int i=;i<=m;i++)
for(int j=h[i];j;j=e[j].net)
if(!e[j].cap&&e[j].to!=)printf("%d %d\n",i,e[j].to);
}
return ;
}

P2756 飞行员配对方案问题(网络流24题之一)的更多相关文章

  1. P2756 飞行员配对方案问题 网络流

    P2756 飞行员配对方案问题 #include <bits/stdc++.h> using namespace std; , inf = 0x3f3f3f; struct Edge { ...

  2. 洛谷 [P2756] 飞行员配对方案问题 网络流实现

    网络流实现二分图匹配 对于x集合的每一个点连一条从源点出发的容量为一的边,对于y集合的每一个点连一条到汇点的容量为一的边,跑最大流 #include <iostream> #include ...

  3. 洛谷P2756 飞行员配对方案问题 网络流_二分图

    Code: #include<cstdio> #include<queue> #include<vector> #include<cstring> #i ...

  4. 洛谷 P2756 飞行员配对方案问题 (二分图/网络流,最佳匹配方案)

    P2756 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其 ...

  5. 洛谷P2756飞行员配对方案问题 P2055假期的宿舍【二分图匹配】题解+代码

    洛谷 P2756飞行员配对方案问题 P2055假期的宿舍[二分图匹配] 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架 ...

  6. 洛谷——P2756 飞行员配对方案问题

    P2756 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其 ...

  7. 洛谷 P2756 飞行员配对方案问题 (二分图匹配)

    题目链接:P2756 飞行员配对方案问题 题意 给定 \(m\) 个外籍飞行员和 \(n - m\) 个英国飞行员,每一架飞机需要一名英国飞行员和一名外籍飞行员,求最多能派出几架飞机. 思路 最大流 ...

  8. luogu P2756 飞行员配对方案问题

    题目链接:P2756 飞行员配对方案问题 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其中1 名是英国飞行员,另 ...

  9. 洛谷P2756 飞行员配对方案问题(二分图匹配)

    P2756 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其 ...

随机推荐

  1. 【LG3703】[SDOI2017]树点涂色

    [LG3703][SDOI2017]树点涂色 题面 洛谷 题解 更博辣,更博辣!!! 猪年的第一篇博客 一次只能染根到\(x\),且染的颜色未出现过 这句话是我们解题的关键. 设\(x\)到根的颜色数 ...

  2. 简单读取 properties文件

    看了网上很多读取的方法,都太过复杂,直接使用下面的方法就可以简单读取 properties文件了 ide使用idea 测试读取成功 import java.util.ResourceBundle; p ...

  3. 国外10个ASP.Net C#下的开源CMS

    国外10个ASP.Net C#下的开源CMS https://blog.csdn.net/peng_hai_lin/article/details/8612895   1.Ludico Ludico是 ...

  4. MyBatis.Net 配置

    假设我们现在有这样的需求,要对学生信息进行管理 学生表有要以下要求 字段名称 数据类型 说明 stuNo 字符 学号,该列必填,为主键递增 stuName 字符 学生姓名,该列必填,要考虑姓氏可能是两 ...

  5. Selenium2+python自动化-CSS定位语法

    前言 一些人在使用selenium定位元素时,用的是xpath定位,因为xpath基本能解决定位的需求.css定位往往被忽略掉了,其实css定位也有它的价值,css定位更快,语法更简洁.这一篇css的 ...

  6. Django-建立网页

    进入cmd模式做 django-admin startproject helloworld创建一个project,并命名helloworld,新生成的文件结构如下   输入python manage. ...

  7. 【UGUI】 (一)------- 放大镜

    在许多游戏或应用中,我们常常看到放大镜的身影,而在Unity里面,制作一个简易的放大镜是非常简单的.    一. 创建一个3DObject 创建一个Cube或者 Cylinder,这里为了更像放大镜一 ...

  8. 实现短信超链接调起APP

    因APP推广的需求,需要给APP用户定期发送短信提醒登录使用,为了更好的用户体验在短信内容中嵌入了可以直接打开APP的超链接,下面介绍一下具体的代码实现. 编辑openApp.html文件: < ...

  9. File Transfer(并查集)

    题目大意:将多个电脑通过网线连接起来,不断查询2台电脑之间是否连通. 问题来源:中国大学mooc 05-树8 File Transfer (25 分) We have a network of com ...

  10. 基于C#的机器学习--机器学习的基本知识

    机器学习的基本知识 ,…用n个观测值测量.但我们不再对Y的预测感兴趣,因为我们不再有Y了,我们唯一感兴趣的是在已有的特征上发现数据模式: ​ 在前面的图中,我们可以看到这样的数据本身更适合于非线性方法 ...