路径覆盖就是在图中找一些路径,使之覆盖了图中的所有顶点,且任何一个顶点有且只有一条路径与之关联。

对于一个有向无环图怎么求最小路径覆盖?

先构造二分图: 对于原图,先拆点,吧每个点i拆成ii,iii。若有边i--》j,则在二分图中,添加边 ii--》jjj(即原来每个点拆为一个入点和出点),这样构成二分图。

则:最小路径覆盖数=原图顶点数-二分图最大匹配数。

粗略解析证明:(设有n个顶点)

若原图没有边,则最大匹配数为0,最小路径覆盖为n,思想:每得到一个匹配,相当于把这俩个点并为一个集合(原来有N个集合),即这俩个点在原图中是在同一条路径覆盖上的,每次成功匹配,相当于一次成功“并集”,所谓的路径覆盖,可以理解为合并顶点的动作,而匹配的点不重复(分出俩个点恰好对应路径覆盖时该店的一出一入),每成功一次匹配,则顶点集合少了一,即路径少了一条,所以最小路径覆盖对应最大匹配的时候,即证。

该题(hdu3861),题意:划分一个有向图,要求:1,:同一个强连通分量(SCC)中的点属于一个集合,2:每个点只属于一个集合。3:任意俩个点都可以单向抵达的(不能越过其他集合)是属于一个集合。

原图不是无环图,先缩点,成有向无环图,在求最下路径覆盖,按上面方法,代码有注解:

(转载请注明出处。)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<queue>
#include<stack>
using namespace std;
int n,m;
const int MAX=50010*2;const int inf=0x3f3f3f3f;
vector<vector<int> >v(MAX);
int vis[MAX];int dfn[MAX];int low[MAX];
int times=0;int scc[MAX];int ins[MAX];stack<int>s;
int num=0; //缩点后点数
int e[MAX*3][3];int head[MAX*2];int nume=0;
void clea() //初始化工作
{
for(int i=0;i<=2*n+1;i++)
{
head[i]=-1;
ins[i]=dfn[i]=vis[i]=low[i]=scc[i]=0;
v[i].clear();
}
num=0;times=0;nume=0;
}
void inline adde(int f,int s,int w) //添加新图(二分图)的边,用网络流法解,所用之
{
e[nume][0]=s;e[nume][1]=head[f];head[f]=nume;
e[nume++][2]=w;
e[nume][0]=f;e[nume][1]=head[s];head[s]=nume;
e[nume++][2]=0;
}
void tarjan(int u) //有向图缩点
{
dfn[u]=low[u]=++times;
ins[u]=1;
s.push(u);
for(int i=0;i<v[u].size();i++)
{
int ch=v[u][i];
if(!vis[ch])
{
vis[ch]=1;
tarjan(ch);
if(low[ch]<low[u])
low[u]=low[ch];
}
else
if(ins[ch]&&dfn[ch]<low[u])
low[u]=dfn[ch];
}
if(low[u]==dfn[u])
{
int cur;
num++;
do
{
cur=s.top();
s.pop();
scc[cur]=num;
ins[cur]=0;
}while(cur!=u);
}
}
void get_newgraph() //获得新图,二分图
{
for(int i=1;i<=n;i++)
{
for(int j=0;j<v[i].size();j++)
{
int c=v[i][j];
if(scc[i]!=scc[c])
{
adde(scc[i],scc[c]+num,1);
}
}
}
for(int i=1;i<=num;i++)
{
adde(0,i,1);
adde(i+num,num+num+1,1);
}
}
int lev[MAX]; //网络流法求最大匹配 添加源汇点(0,2*num+1),流量为1,有重边也不无纺。
bool bfs()
{
for(int i=0;i<=num*2+1;i++)
{
lev[i]=vis[i]=0;
}
queue<int>q;
q.push(0);vis[0]=1;
while(!q.empty())
{
int cur=q.front();
q.pop();
for(int i=head[cur];i!=-1;i=e[i][1])
{
int c=e[i][0];
if(!vis[c]&&e[i][2]>0)
{
lev[c]=lev[cur]+1;
if(c==num*2+1)return 1;
vis[c]=1;
q.push(c);
}
}
}
return vis[num*2+1];
}
int dfs(int u,int minf)
{
if(u==num*2+1||minf==0)return minf;
int sum=0,f;
for(int i=head[u];i!=-1&&minf;i=e[i][1])
{
int c=e[i][0];
if(lev[c]==lev[u]+1&&e[i][2]>0)
{
f=dfs(c,minf<e[i][2]?minf:e[i][2]);
e[i][2]-=f;e[i^1][2]+=f;
sum+=f;minf-=f;
}
}
return sum;
}
int dinic()
{
int sum=0;
while(bfs())
{
sum+=dfs(0,inf);
}
return sum;
}
int main()
{
int ta;
scanf("%d",&ta);
while(ta--)
{
scanf("%d%d",&n,&m);
int aa,bb;
clea();
for(int i=0;i<m;i++)
{
scanf("%d%d",&aa,&bb);
v[aa].push_back(bb);
}
for(int i=1;i<=n;i++)
{
if(!vis[i])
{
vis[i]=1;
tarjan(i);
}
}
get_newgraph();
int ans=num-dinic(); //最小路径覆盖数
printf("%d\n",ans);
}
return 0;
}

有向图最小路径覆盖方法浅析、证明 //hdu 3861的更多相关文章

  1. HDOJ1151有向图最小路径覆盖

    //有向图最小路径覆盖:从某一点出发沿着有向路径,不走回路,能将所有的结点遍历. #include<iostream> #include<cstdio> #include< ...

  2. (匹配 最小路径覆盖)Air Raid --hdu --1151

    链接: http://acm.hdu.edu.cn/showproblem.php?pid=1151 http://acm.hust.edu.cn/vjudge/contest/view.action ...

  3. ●hihocoder #1394 网络流四·最小路径覆盖

    题链: http://hihocoder.com/problemset/problem/1394 题解: 有向图最小路径覆盖:最少的路径条数不重不漏的覆盖所有点. 注意到在任意一个最小路径覆盖的方案下 ...

  4. HDU 3861 The King’s Problem 最小路径覆盖(强连通分量缩点+二分图最大匹配)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3861 最小路径覆盖的一篇博客:https://blog.csdn.net/qq_39627843/ar ...

  5. hdu 1151 Air Raid(二分图最小路径覆盖)

    http://acm.hdu.edu.cn/showproblem.php?pid=1151 Air Raid Time Limit: 1000MS   Memory Limit: 10000K To ...

  6. (step6.3.4)hdu 1151(Air Raid——最小路径覆盖)

    题意:     一个镇里所有的路都是单向路且不会组成回路. 派一些伞兵去那个镇里,要到达所有的路口,有一些或者没有伞兵可以不去那些路口,只要其他人能完成这个任务.每个在一个路口着陆了的伞兵可以沿着街去 ...

  7. HDU 3861 The King’s Problem(tarjan连通图与二分图最小路径覆盖)

    题意:给我们一个图,问我们最少能把这个图分成几部分,使得每部分内的任意两点都能至少保证单向连通. 思路:使用tarjan算法求强连通分量然后进行缩点,形成一个新图,易知新图中的每个点内部的内部点都能保 ...

  8. HDU 3861.The King’s Problem 强联通分量+最小路径覆盖

    The King’s Problem Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...

  9. HDU 3861 The King’s Problem(强连通+二分图最小路径覆盖)

    HDU 3861 The King's Problem 题目链接 题意:给定一个有向图,求最少划分成几个部分满足以下条件 互相可达的点必须分到一个集合 一个对点(u, v)必须至少有u可达v或者v可达 ...

随机推荐

  1. 利用python进行数据分析1_numpy的基本操作,建模基础

    import numpy as np # 生成指定维度的随机多维数据 data=np.random.rand(2,3) print(data) print(type(data)) 结果: [[0.11 ...

  2. WINDOWS-基础:WINDOWS常用API

    1.窗口信息 //MS 为我们提供了打开特定桌面和枚举桌面窗口的函数. hDesk=OpenDesktop(lpszDesktop,,FALSE,DESKTOP_ENUMERATE); //打开我们默 ...

  3. javase(8)_集合框架_List、Set、Map

    一.集合体系(不包括Queue体系) 二.ArrayList ArrayList的属性 private transient Object[] elementData; //存储元素 private i ...

  4. ViewController的lifecycle和autolayout

  5. python常用内置函数用法精要

    用一个表格大致总结一下所有的内置函数用法,如下: 函数 功能简要说明 abs(x) 返回数字x的绝对值或复数x的模 all(iterable) 如果对于可迭代对象中所有元素x都等价于True,则返回T ...

  6. centos7内核优化

    #sysctl -p 参数: net.ipv6.conf.all.disable_ipv6 = 1net.ipv6.conf.default.disable_ipv6 = 1net.ipv4.icmp ...

  7. Python基本运算符和流程控制

    常量 常量即不可改变的量,在Python中不存在常量,我们只能逻辑上规定一个常量并不去修改它,通常用全大写字母表示. 基本运算符之二 算术运算 运算符 说明 ** 幂运算 *, /, //, % 乘. ...

  8. LeetCode(2)Add Two Numbers

    题目: You are given two linked lists representing two non-negative numbers. The digits are stored in r ...

  9. python库——h5py入门讲解

    本文只是简单的对h5py库的基本创建文件,数据集和读取数据的方式进行介绍,作者刚接触h5py,完全靠看文档自学,如果哪里说的不对,欢迎纠正!如果读者需要进一步详细的学习h5py的更多知识,请参考h5p ...

  10. android sdk 下载

    不知道是因为最近kaihui还是怎么的,打开android sdk官方网站特别的慢,想下载最新版本的platform几乎变成不可能完成的任务,不知道为什么Google不像Apache那样在各国设立镜像 ...