zoukankan      html  css  js  c++  java
  • 【CJOJ1494】【洛谷2756】飞行员配对方案问题

    题面

    题目背景

    第二次世界大战时期..

    题目描述

    英国皇家空军从沦陷国征募了大量外籍飞行员。由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的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

    题解

    CJOJ和洛谷上的两道题的题面略有不同
    但是所求的东西是差不多的。
    因此直接放在一起。

    很明显
    现在飞行员分为了两组
    并且不存在同一组内的连线
    这是一个二分图的最大匹配问题

    直接使用匈牙利算法即可求解。

    大佬们说这道题可以使用网络流最大流来做,但是本蒟蒻暂时还不会
    以后再进行网络流方法的补充

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<cmath>
    #include<algorithm>
    using namespace std;
    #define MAX 200
    #define MAXL 200*200
    struct Line
    {
    	  int v,next;
    }e[MAXL];
    int h[MAX],cnt=0;
    int sum=0;
    int match[MAX];
    int n,m,u,v;
    bool vis[MAX];
    inline void Add(int u,int v)
    {
    	   e[cnt]=(Line){v,h[u]};
    	   h[u]=cnt++;
    }
    bool DFS(int u)
    {
    	    for(int i=h[u];i!=-1;i=e[i].next)
    	    {
    	    	    int v=e[i].v;
    	    	    if(!vis[v])
    	    	    {
    	    	    	   vis[v]=true;
    	    	    	   if(!match[v]||DFS(match[v]))
    	    	    	   {
    	    	    	   	       match[v]=u;
    	    	    	   	       return true;
    	    	    	   }
    	    	    }
    	    }
    	    return false;
    }
    int main()
    {
    	memset(h,-1,sizeof(h));
        cin>>n>>m;
        while(true)
        {
        	  cin>>u>>v;
        	  if(u==-1)break;
        	  Add(u,v);
        }
        for(int i=1;i<=n;++i)
        {
        	   memset(vis,0,sizeof(vis));
        	   if(DFS(i))++sum;
        }
        cout<<sum<<endl;
        for(int i=n+1;i<=m;++i)
        {
        	    if(match[i])
        	    {
        	    	   cout<<match[i]<<' '<<i<<endl;
        	    }
        }
        return 0;
    }
    
    
  • 相关阅读:
    裸眼 3D 技术是什么原理?
    裸眼3D全攻略3:拍摄3D—瞳距、镜距、视角偏转与空间感
    JFreeChart DateAxis用法
    remount issue on android 7.0
    获取WebView加载的网页内容并进行动态修改
    android自定义Activity窗口大小(theme运用)
    Android5.0免Root截屏,录屏
    coursera上的软件安全课程的课后阅读补充
    java,C#接口与C++的虚基类
    单元测试之C/C++
  • 原文地址:https://www.cnblogs.com/cjyyb/p/7197222.html
Copyright © 2011-2022 走看看