训练指南 UVA - 11419(二分图最小覆盖数)
layout:     post
title:      训练指南 UVA - 11419(二分图最小覆盖数)
author:     "luowentaoaa"
catalog:  true
mathjax: true
tags:
- 二分图
- 最小点覆盖
- 图论
- 训练指南
SAM I AM
题目大意:给出一个R×C的网格,网格上棉纺了一些目标。可以在网格外发射子弹,子弹会沿着垂直或水平方向飞行,并且打掉飞行路径上的所有目标。你的任务是计算出最少需要多少子弹,各从哪个位置发射,才能把所有目标全部打掉。
解题思路:König定理:最小覆盖数等于最大匹配数。把目标所在的坐标,转化为XY结点,行看成X结点,列看成Y结点。那现在问题就变成了,如何选最少的结点,覆盖所有的边。
求最小覆盖的步骤大致如下:1)在右边找到一个未被匹配过的点,标记。2)走一条没被匹配过的边,到左边的点,标记。3)走一条匹配过的边到右边,标记。4)重复2,3步骤直到不能再走。5)回到步骤一,直到找不到未被匹配且未被标记的右边的点。6)标记结束后,右边没有标记的点,和左边标记过的点,就可以覆盖所有的边。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll mod=998244353;
const int maxn=1e3+50;
const ll inf=1e10;
const ll INF = 1000000000;
const double eps=1e-5;
#define bug cout<<"bbibibibbbb="<<endl;
/// 二分图最大基数匹配
struct BPM{
    int n,m;    /// 左右顶点个数
    vector<int>G[maxn]; /// 邻接表
    int left[maxn];    /// left[i]为右边第i个点的匹配点编号,-1表示不存在
    bool T[maxn];       /// T[i]为右边第i个点是否已标记
    int right[maxn];        /// 求最小覆盖用
    bool S[maxn];           /// 求最小覆盖用
    void init(int n,int m){
        this->n=n;
        this->m=m;
        for(int i=0;i<n;i++)G[i].clear();
    }
    void AddEdge(int u,int v){
        G[u].push_back(v);
    }
    bool match(int u){
        S[u]=true;
        for(int i=0;i<G[u].size();i++){
            int v=G[u][i];
            if(!T[v]){
                T[v]=true;
                if(left[v]==-1||match(left[v])){
                    left[v]=u;
                    right[u]=v;
                    return true;
                }
            }
        }
        return false;
    }
    /// 求最大匹配
    int solve(){
        memset(left,-1,sizeof(left));
        memset(right,-1,sizeof(right));
        int ans=0;
        for(int u=0;u<n;u++){
            memset(S,0,sizeof(S));
            memset(T,0,sizeof(T));
            if(match(u))ans++;
        }
        return ans;
    }
    /// 求最小覆盖。X和Y为最小覆盖中的点集
    int mincover(vector<int>& X,vector<int>& Y){
        int ans=solve();
        memset(S,0,sizeof(S));
        memset(T,0,sizeof(T));
        for(int u=0;u<n;u++)
            if(right[u]==-1)match(u);
        for(int u=0;u<n;u++)
            if(!S[u])X.push_back(u);
        for(int v=0;v<n;v++)
            if(T[v])Y.push_back(v);
        return ans;
    }
};
BPM solver;
int R,C,N;
int main()
{
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    std::cout.tie(0);
    int kase=0;
    while(cin>>R>>C>>N&&R&&C&&N){
        solver.init(R,C);
        for(int i=0;i<N;i++){
            int r,c;
            cin>>r>>c;r--;c--;
            solver.AddEdge(r,c);
        }
        vector<int>X,Y;
        int ans=solver.mincover(X,Y);
        cout<<ans;
        for(int i=0;i<X.size();i++)cout<<" r"<<X[i]+1;
        for(int j=0;j<Y.size();j++)cout<<" c"<<Y[j]+1;
        cout<<endl;
    }
    return 0;
}
训练指南 UVA - 11419(二分图最小覆盖数)的更多相关文章
- 训练指南 UVA - 11383(KM算法的应用 lx+ly >=w(x,y))
		layout: post title: 训练指南 UVA - 11383(KM算法的应用 lx+ly >=w(x,y)) author: "luowentaoaa" cata ... 
- 训练指南 UVA - 11354(最小生成树 + 倍增LCA)
		layout: post title: 训练指南 UVA - 11354(最小生成树 + 倍增LCA) author: "luowentaoaa" catalog: true ma ... 
- 训练指南 UVA - 11090(最短路BellmanFord+ 二分判负环)
		layout: post title: 训练指南 UVA - 11090(最短路BellmanFord+ 二分判负环) author: "luowentaoaa" catalog: ... 
- 训练指南 UVA - 11374(最短路Dijkstra + 记录路径 + 模板)
		layout: post title: 训练指南 UVA - 11374(最短路Dijkstra + 记录路径 + 模板) author: "luowentaoaa" catalo ... 
- 训练指南 UVA - 11478(最短路BellmanFord+ 二分+ 差分约束)
		layout: post title: 训练指南 UVA - 11478(最短路BellmanFord+ 二分+ 差分约束) author: "luowentaoaa" catal ... 
- 训练指南 UVA - 10917(最短路Dijkstra + 基础DP)
		layout: post title: 训练指南 UVA - 10917(最短路Dijkstra + 基础DP) author: "luowentaoaa" catalog: tr ... 
- 训练指南 UVA - 11324(双连通分量 + 缩点+ 基础DP)
		layout: post title: 训练指南 UVA - 11324(双连通分量 + 缩点+ 基础DP) author: "luowentaoaa" catalog: true ... 
- 算法竞赛入门经典训练指南——UVA 11300  preading the Wealth
		A Communist regime is trying to redistribute wealth in a village. They have have decided to sit ever ... 
- UVa 1349 (二分图最小权完美匹配) Optimal Bus Route Design
		题意: 给出一个有向带权图,找到若干个圈,使得每个点恰好属于一个圈.而且这些圈所有边的权值之和最小. 分析: 每个点恰好属于一个有向圈 就等价于 每个点都有唯一后继. 所以把每个点i拆成两个点,Xi ... 
随机推荐
- WIN7服务优化,别关太多,小心启动不
			原文链接地址:http://blog.csdn.net/civilman/article/details/51423972 Adaptive brightness 监视周围的光线状况来调节屏幕明暗,如 ... 
- iOS 屏幕适配,autoResizing autoLayout和sizeClass图文详解
			=== 1. autoResizing autoresizing是苹果早期的ui布局适配的解决办法,iOS6之前完全可以胜任了,因为苹果手机只有3.5寸的屏幕,在加上手机app很少支持横屏,所以iOS ... 
- Any gotchas at all with converting from MyISAM to InnoDB?
			Q: I'm ready to move from MyISAM to InnoDB but wanted to know if there was a full list of things to ... 
- HDU2732:Leapin' Lizards(最大流)
			Leapin' Lizards Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)T ... 
- es6+最佳入门实践(5)
			5.对象扩展 5.1.对象简写 在es5中,有这样一种写法 var name = "xiaoqiang"; var age = 12; var obj = { name : nam ... 
- codevs1163访问艺术馆 树形dp
			算裸的树形dp吧 回来复习一波 #include<cstdio> #include<cstring> #include<algorithm> #include< ... 
- BZOJ1082_栅栏_C++
			题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1082 题解:http://www.cnblogs.com/hadilo/p/5924546.h ... 
- mobius反演讲解
			mobius反演的基本形式为,假设知道函数F(x)=Σf(d) d|x,那么我们可以推出f(x)=Σmiu(d)*F(x/d) d|x,另一基本形式为假设知道函数F(x)=Σf(d) x|d,那么我们 ... 
- python自带的线程池和进程池
			#python自带的线程池 from multiprocessing.pool import ThreadPool #注意ThreadPool不在threading模块下 from multiproc ... 
- linux基础 -nginx和nfs代理 开发脚本自动部署及监控
			开发脚本自动部署及监控 1.编写脚本自动部署反向代理.web.nfs: (1).部署nginx反向代理三个web服务,调度算法使用加权轮询: (2).所有web服务使用共享存储nfs,保证所有web ... 
