http://poj.org/problem?id=2195

题意 :  N*M的点阵中,有N个人,N个房子。让x个人走到这x个房子中,只能上下左右走,每个人每走一步就花1美元,问当所有的人都归位了之后,需要花多少美元。

思路 :最小费用最大流。把人作为一个顶点集合U,房子作为另一个顶点集合V,把U中所有点到V中所有点连线,费用cost[u][v]为abs(△x)+abs(△y),反向弧费用cost[v][u]= -cost[u][v],容量cap[u][v]=1,构成一个多源多汇的二分图。 由于每一个多源多汇的网络流都必有一个与之对应的单源单汇的网络流,为了便于解题,由此构造一个超级源s和超级汇t,超级源s与U中所有点相连,费用cost[s][u]=0(这是显然的),容量cap[s][u]=1;V中所有点与超级汇t相连,费用cost[v][t]=0(这是显然的),容量cap[t][v]=1。至于其他不连通的点,费用与容量均为0。容量为0的边,可以理解为饱和边,不再连通。而上述的所有边之所以容量初始化为1,是因为每间房子只允许入住1个人。而与超级源(汇)相连的边的费用之所以为0,是为了现在所构造的单源单汇网络流最终所求的最小费用等于原来的多源多汇网络流的最小费用。

#include <iostream>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <queue>
#include <math.h> using namespace std; const int maxn = ;
struct node
{
int u ;
int v ;
}p[],h[] ; int pre[maxn],dist[maxn],n,m,pn,hn,s,t,cnt;
int cap[maxn][maxn],flow[maxn][maxn],cost[maxn][maxn] ;
bool flag[maxn] ;
char ch[maxn][maxn] ; const int INF = ; void Init()
{
pn = hn = ;
cnt = s = ;
memset(cap,,sizeof(cap)) ;
memset(flow,,sizeof(flow)) ;
memset(cost,,sizeof(cost)) ;
}
void spfa()
{
queue<int>Q ;
for(int i = ; i < maxn ; i++)
dist[i] = INF ;
memset(pre,-,sizeof(pre)) ;
memset(flag,false,sizeof(flag)) ;
Q.push(s) ;
flag[s] = true ;
dist[s] = ;
while(!Q.empty())
{
int u = Q.front() ;
Q.pop() ;
flag[u] = false ;
for(int v = ; v <= t ; v++)
{
if(cap[u][v] && dist[v] > dist[u] + cost[u][v])
{
dist[v] = dist[u] + cost[u][v] ;
pre[v] = u ;
if(!flag[v])
{
Q.push(v) ;
flag[v] = true ;
}
}
}
}
} void mcmf()
{
for( ; ; )
{
spfa() ;
if(pre[t] == -) break ;//没有父节点了,
int x = t ,minn = INF ;
while(pre[x] != -)
{
minn = min(minn,cap[pre[x]][x]) ;
x = pre[x] ;
}
x = t ;
while(pre[x] != -)
{
cap[pre[x]][x] -= minn ;
cap[x][pre[x]] += minn ;
cnt += minn*cost[pre[x]][x];
x = pre[x];
}
} } int main()
{
while(scanf("%d %d",&n,&m) != EOF)
{
if(n == && m == ) break ;
Init() ;
for(int i = ; i < n ; i++)
{
scanf("%s",ch[i]) ;
for(int j = ; j < m ; j++)
{
if(ch[i][j] == 'H')
{
h[++hn].u = i ;
h[hn].v = j ;
}
else if(ch[i][j] == 'm')
{
p[++pn].u = i ;
p[pn].v = j ;
}
}
}
t = pn+hn+ ;
for(int i = ; i <= pn ; i++)
cap[s][i] = ;
for(int i = ; i <= hn ; i++)
cap[i+pn][t] = ;
for(int i = ; i <= pn ; i++)
{
for(int j = ; j <= hn ; j++)
{
cap[i][j+pn] = ;
cost[i][j+pn] = fabs(p[i].u-h[j].u) + fabs(p[i].v-h[j].v) ;
cost[j+pn][i] = -cost[i][j+pn] ;
}
}
mcmf() ;
printf("%d\n",cnt) ;
}
return ;
}

POJ 2195 Going Home(最小费用最大流)的更多相关文章

  1. POJ 2195 - Going Home - [最小费用最大流][MCMF模板]

    题目链接:http://poj.org/problem?id=2195 Time Limit: 1000MS Memory Limit: 65536K Description On a grid ma ...

  2. POJ 2195 Going Home 最小费用最大流 尼玛,心累

    D - Going Home Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Subm ...

  3. poj 2195 Going Home(最小费用最大流)

    题目:http://poj.org/problem?id=2195 有若干个人和若干个房子在一个给定网格中,每人走一个都要一定花费,每个房子只能容纳一人,现要求让所有人进入房子,且总花费最小. 构造一 ...

  4. poj 2351 Farm Tour (最小费用最大流)

    Farm Tour Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 17230   Accepted: 6647 Descri ...

  5. POJ 2157 Evacuation Plan [最小费用最大流][消圈算法]

    ---恢复内容开始--- 题意略. 这题在poj直接求最小费用会超时,但是题意也没说要求最优解. 根据线圈定理,如果一个跑完最费用流的残余网络中存在负权环,那么顺着这个负权环跑流量为1那么会得到更小的 ...

  6. poj 2135 Farm Tour 最小费用最大流建图跑最短路

    题目链接 题意:无向图有N(N <= 1000)个节点,M(M <= 10000)条边:从节点1走到节点N再从N走回来,图中不能走同一条边,且图中可能出现重边,问最短距离之和为多少? 思路 ...

  7. POJ 3680: Intervals【最小费用最大流】

    题目大意:你有N个开区间,每个区间有个重量wi,你要选择一些区间,使得满足:每个点被不超过K个区间覆盖的前提下,重量最大 思路:感觉是很好想的费用流,把每个区间首尾相连,费用为该区间的重量的相反数(由 ...

  8. POJ 2135 Farm Tour [最小费用最大流]

    题意: 有n个点和m条边,让你从1出发到n再从n回到1,不要求所有点都要经过,但是每条边只能走一次.边是无向边. 问最短的行走距离多少. 一开始看这题还没搞费用流,后来搞了搞再回来看,想了想建图不是很 ...

  9. [poj] 1235 Farm Tour || 最小费用最大流

    原题 费用流板子题. 费用流与最大流的区别就是把bfs改为spfa,dfs时把按deep搜索改成按最短路搜索即可 #include<cstdio> #include<queue> ...

  10. POJ 2516 Minimum Cost [最小费用最大流]

    题意略: 思路: 这题比较坑的地方是把每种货物单独建图分开算就ok了. #include<stdio.h> #include<queue> #define MAXN 500 # ...

随机推荐

  1. JS获取网页宽高方法集合

    JS获取网页宽高等方法的集合:document.body.clientWidth - 网页可见区域宽document.body.clientHeight - 网页可见区域高 document.body ...

  2. postgresql 将同一个字段的值组合和将多个字段的值组合

    多字段值根据连接符拼接 concat_ws(':',aaa,bbb) 单字段值根据连接符拼接 string_agg(ccc,' \r\n ') 如果要将多个字段的值拼接成一个: string_agg( ...

  3. 20160509-hibernate-集合映射

    集合映射 集合映射(set, list, array,bag, map) <set name=”employees” > <key column=”depart_id”/> & ...

  4. 新建的linux虚拟机找不到eth0解决办法

    新建的linux虚拟机通过 vi /etc/sysconfig/network-scripts/ifcfg-eth0进行配置: 再用ifconfig查看ip竟然没有eth0,只有lo和virbr0 其 ...

  5. Java内存溢出的详细解决方案

    本文介绍了Java内存溢出的详细解决方案.本文总结内存溢出主要有两种情况,而JVM经常调用垃圾回收器解决内存堆不足的问题,但是有时仍会有内存不足的错误.作者分析了JVM内存区域组成及JVM设置虚拟内存 ...

  6. 【HTTPS】Https和SSL学习笔记(二)

    此文讲述证书的相关信息,参考文章链接http://www.guokr.com/post/116169/ 一. 证书的类型 常用的几种证书如下: (1) SSL证书,用于加密HTTP (2) 代码签名证 ...

  7. Jquery 1.11.1 Checkbox checked 判断

    if($(this).prop('checked'))

  8. ###Canny边缘检测算子

    开源中国. #@date: 2014-06-20 #@author: gerui #@email: forgerui@gmail.com 一.一阶微分边缘算子 1. 一阶微分边缘检测算子也称梯度边缘算 ...

  9. java包和jar包

    1.包 package  pack;                             /*定义包,放在程序的第一行,包名所以字母小写*/ class PackageDemo{    publi ...

  10. bzoj 1066: [SCOI2007] 蜥蜴

    这道题还是挺好想的,但我一开始还是想错了…… 把每个石柱拆成两个点,一个入度,一个出度,两个点连一条容量为高度的边,这样就可以限制从此石柱上经过的蜥蜴的数量.关于蜥蜴是否单独成点,我是单独当成了一个点 ...