zoukankan      html  css  js  c++  java
  • hdu 2119(最大匹配)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2119

    题意很好理解,就是每次可以删除一行或者一列数,问最少几次可以把所有的1都变成0,也就是都删完。用二分图表示,行代表二分图的一部分,列代表二分图的一部分,map[i][j]==1代表连一条边,这样就转化为求最小顶点覆盖,即求二分图的最大匹配(边数最多的匹配,即把尽可能多的边与某一个顶点相关联,这样选择全部的边所需要的最少顶点就是最小顶点覆盖)。此外,二分图还有最小路径覆盖,意思是用最少的边把图中所有的顶点都遍历到(最小路径覆盖 = 顶点数 - 最大匹配)。

    View Code
     1 #include<iostream>
     2 #include<cstring>
     3 const int N=110;
     4 using namespace std;
     5 int map[N][N];
     6 int visited[N];
     7 int used[N];
     8 int n,m;
     9 
    10 //从定点x出发,用深度优先搜索策略寻找增广路
    11 int Solve(int x){
    12     for(int j=0;j<m;j++){
    13         if(map[x][j]&&!visited[j]){
    14             visited[j]=1;
    15             //如果没有匹配,或者已经匹配了,但从used[j]出发可以找到一天增广路;
    16             //如果前一个条件成立,则不会递归调用
    17             if(used[j]==-1||Solve(used[j])){
    18                 used[j]=x;
    19                 return 1;
    20             }
    21         }
    22     }
    23     return 0;
    24 }
    25 
    26 
    27 //求二部图最大匹配的匈牙利算法
    28 int MaxMatch(){
    29     int count=0;
    30     for(int i=0;i<n;i++){
    31         memset(visited,0,sizeof(visited));
    32         if(Solve(i))count++;
    33     }
    34     return count;
    35 }
    36 
    37 int main(){
    38     while(~scanf("%d",&n)&&n){
    39         scanf("%d",&m);
    40         for(int i=0;i<n;i++){
    41             for(int j=0;j<m;j++){
    42                 scanf("%d",&map[i][j]);
    43             }
    44         }
    45         memset(used,-1,sizeof(used));
    46         int ans=MaxMatch();
    47         printf("%d\n",ans);
    48     }
    49     return 0;
    50 }
  • 相关阅读:
    增加增删改查按钮
    基于.net创建一份报表模块
    bootstrap 三层设计
    DI 依赖注入之StructureMap框架
    unit vs2017基于nunit framework创建单元测试
    Oracle彻底卸载
    sql 统计常用的sql
    Webserver asp配置及伪静态设置
    MVC ASP.NET MVC5使用Area区域
    VS 发布MVC网站缺少视图解决方案
  • 原文地址:https://www.cnblogs.com/wally/p/2966638.html
Copyright © 2011-2022 走看看