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. Each child in an array or iterator should have a unique "key" prop. Check the render method of `CreditCategoryModal`

    参考地址:http://f00sun.com/category/react

  2. Spark与Hadoop计算模型的比较分析

    http://tech.it168.com/a2012/0401/1333/000001333287.shtml 最近很多人都在讨论Spark这个貌似通用的分布式计算模型,国内很多机器学习相关工作者都 ...

  3. ACM——第几天

    第几天 时间限制(普通/Java) : 1000 MS/ 3000 MS          运行内存限制 : 65536 KByte总提交 : 1830            测试通过 : 525 描 ...

  4. VS2015+AngularJS+Ionic开发

    安装VS2015 Update2的过程是非常曲折的.还好经过不懈的努力,终于折腾成功了. 如果开发Cordova项目的话,推荐大家用一下ionic这个框架,效果还不错.对于Cordova.PhoneG ...

  5. JAVA的StringBuffer类(转载整理)____非常重要的一个类,线程安全,不用每次创建一个对象,以及和String的区别

    核心部分转载自:http://www.cnblogs.com/springcsc/archive/2009/12/03/1616330.html StringBuffer类和String一样,也用来代 ...

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

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

  7. Oracle Split Partitions

    1. 创建分离分区的存储过程 CREATE OR REPLACE Procedure SP_Split_Partition( v_table_name_in in varchar2, v_part_n ...

  8. UIMenuController/UIPasteboard(1) 制作一个可以粘贴复制的Label

    效果如下:   苹果只放出来了 UITextView,UITextField,webView三个控件的剪贴板,所以我们要自定义可以复制粘贴的控件,首先需要打开UIResponder的两个方法: - ( ...

  9. jquery个人笔记

    一.链式操作 <!DOCTYPE html> <html> <head> <title></title> <script src = ...

  10. Cabarc Overview (Microsoft TechNet)

    Original Link:  Cabarc Overview Applies To: Windows Server 2003, Windows Server 2003 R2, Windows Ser ...