POJ 2195 Going Home (费用流)
题面
On a grid map there are n little men and n houses. In each unit time, every little man can move one unit step, either horizontally, or vertically, to an adjacent point. For each little man, you need to pay a $1 travel fee for every step he moves, until he enters a house. The task is complicated with the restriction that each house can accommodate only one little man.
Your task is to compute the minimum amount of money you need to pay in order to send these n little men into those n different houses. The input is a map of the scenario, a '.' means an empty space, an 'H' represents a house on that point, and am 'm' indicates there is a little man on that point.
You can think of each point on the grid map as a quite large square, so it can hold n little men at the same time; also, it is okay if a little man steps on a grid with a house without entering that house.
Input
There are one or more test cases in the input. Each case starts with a line giving two integers N and M, where N is the number of rows of the map, and M is the number of columns. The rest of the input will be N lines describing the map. You may assume both N and M are between 2 and 100, inclusive. There will be the same number of 'H's and 'm's on the map; and there will be at most 100 houses. Input will terminate with 0 0 for N and M.
Output
For each test case, output one line with the single integer, which is the minimum amount, in dollars, you need to pay.
Sample Input
2 2
.m
H.
5 5
HH..m
.....
.....
.....
mm..H
7 8
...H....
...H....
...H....
mmmHmmmm
...H....
...H....
...H....
0 0
Sample Output
2
10
28
题解
题目大意:给定一张地图,m是人,H是房子,现在每个人都要到一栋房子去,问所有人到房子的曼哈顿距离之和最小是多少?
题解:
可以用KM做
考虑用费用流,
每个人向房子连一条容量为1,费用为曼哈顿距离的边
求最小费用流即可
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<queue>
#include<vector>
using namespace std;
#define MAX 200
#define MAXL 200000
#define INF 1000000000
struct Line
{
int v,next,w,fb,fy;
}e[MAXL];
int ff,hh,mm,ww;
struct Node
{
int x,y;
};
vector<Node> H,M;
int h[MAX*MAX],cnt=1,cost;
int pe[MAX*MAX],pr[MAX*MAX];
inline void Add(int u,int v,int w,int fy)
{
e[cnt]=(Line){v,h[u],w,cnt+1,fy};
h[u]=cnt++;
e[cnt]=(Line){u,h[v],0,cnt-1,-fy};
h[v]=cnt++;
}
int dis[MAX*MAX],S,T,n,m;
char g[MAX][MAX];
bool vis[MAX*MAX];
bool SPFA()
{
memset(dis,63,sizeof(dis));
dis[S]=0;
queue<int> Q;while(!Q.empty())Q.pop();
Q.push(S);
memset(vis,0,sizeof(vis));
while(!Q.empty())
{
int u=Q.front();Q.pop();
vis[u]=false;
for(int i=h[u];i;i=e[i].next)
{
int f=dis[u]+e[i].fy,v=e[i].v;
if(e[i].w&&dis[v]>f)
{
dis[v]=f;
pe[v]=i;
pr[v]=u;
if(!vis[v])
{
vis[v]=true;
Q.push(v);
}
}
}
}
if(dis[T]==dis[T+1])return false;//增广失败
int re=INF;
for(int v=T;v!=S;v=pr[v])
re=min(re,e[pe[v]].w);//计算增广的最大流
for(int v=T;v!=S;v=pr[v])
{
e[pe[v]].w-=re;
e[e[pe[v]].fb].w+=re;
}
ff+=re;
cost+=re*dis[T];
return true;
}
int main()
{
//freopen("POJ2195.in","r",stdin);
while(233)
{
scanf("%d%d",&n,&m);
cnt=1;
memset(h,0,sizeof(h));
if(n==0&&m==0)break;
H.clear();M.clear();hh=mm=cost=ff=0;
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)
{
cin>>g[i][j];
if(g[i][j]=='H')
{
H.push_back((Node){i,j});
hh++;
}
if(g[i][j]=='m')
{
M.push_back((Node){i,j});
mm++;
}
}
S=0;T=mm+hh+1;
for(int i=0;i<M.size();++i)
{
Add(S,i+1,1,0);
for(int j=0;j<H.size();++j)
{
Add(i+1,1+j+mm,1,abs(M[i].x-H[j].x)+abs(M[i].y-H[j].y));//曼哈顿距离
}
}
for(int i=1;i<=hh;++i)
Add(i+mm,T,1,0);
while(SPFA());
printf("%d\n",cost);
}
return 0;
}
POJ 2195 Going Home (费用流)的更多相关文章
- poj - 2195 Going Home (费用流 || 最佳匹配)
http://poj.org/problem?id=2195 对km算法不理解,模板用的也不好. 下面是大神的解释. KM算法的要点是在相等子图中寻找完备匹配,其正确性的基石是:任何一个匹配的权值之和 ...
- POJ 2175 Evacuation Plan (费用流,负环,消圈法,SPFA)
http://poj.org/problem?id=2175 Evacuation Plan Time Limit: 1000MS Memory Limit: 65536K Total Submi ...
- POJ 2516 Minimum Cost (费用流)
题面 Dearboy, a goods victualer, now comes to a big problem, and he needs your help. In his sale area ...
- Going Home POJ - 2195 (最小费用最大流)
On a grid map there are n little men and n houses. In each unit time, every little man can move one ...
- POJ 3680 Intervals(费用流)
Intervals Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 5762 Accepted: 2288 Descrip ...
- POJ 2175 Evacuation Plan 费用流 负圈定理
题目给了一个满足最大流的残量网络,判断是否费用最小. 如果残量网络中存在费用负圈,那么不是最优,在这个圈上增广,增广1的流量就行了. 1.SPFA中某个点入队超过n次,说明存在负环,但是这个点不一定在 ...
- POJ 3680 Intervals(费用流+负权优化)
[题目链接] http://poj.org/problem?id=3680 [题目大意] 有N个带权重的区间,现在要从中选取一些区间, 要求任意点都不被超过K个区间所覆盖,请最大化总的区间权重. [题 ...
- poj 2135 Farm Tour 费用流
题目链接 给一个图, N个点, m条边, 每条边有权值, 从1走到n, 然后从n走到1, 一条路不能走两次,求最短路径. 如果(u, v)之间有边, 那么加边(u, v, 1, val), (v, u ...
- BZOJ3502PA2012Tanie linie&BZOJ2288[POJ Challenge]生日礼物——模拟费用流+链表+堆
题目描述 n个数字,求不相交的总和最大的最多k个连续子序列. 1<= k<= N<= 1000000. 输入 输出 样例输入 5 2 7 -3 4 -9 5 样例输出 13 根据 ...
随机推荐
- mysql 5.7 支持json格式
1.JSON格式的支持:mysql> create table user ( uid int auto_increment, -> data json,primary key(u ...
- mssql学习
1.创建表和数据插入SQL 我们在开始创建数据表和向表中插入演示数据之前,我想给大家解释一下实时数据表的设计理念,这样也许能帮助大家能更好的理解SQL查询. 在数据库设计中,有一条非常重要的规则就是要 ...
- 网卡name-eth1如何修改为eth0
正常来说,Linux在识别网卡时第一张会是eth0,第二张才是eth1. 有时候我们使用虚拟机克隆技术后网卡的信息就会改变,新克隆出来的虚拟主机网卡名字可能变为eth1.无论我们怎么修改都无法改变,这 ...
- EntityFrameWork连接多Db配置
如题所示,EF作为微软主推的ORM工具,最新版本已经是7,说明有很多人在使用它做项目.在使用过程中,可能会连接不同的数据库,本文介绍的是连接SqlServer,MySql和SQLite三种,并且可以互 ...
- 排序算法(Java实现)
这几天一直在看严蔚敏老师的那本<数据结构>那本书.之前第一次学懵懵逼逼,当再次看的时候,发觉写的是非常详细,非常的好. 那就把相关的排序算法用我熟悉的Java语言记录下来了.以下排序算法是 ...
- crontab定时任务(centos)
cron服务是Linux的内置服务,但它不会开机自动启动.可以用以下命令启动和停止服务: /sbin/service crond start /sbin/service crond stop /sbi ...
- Batch Normalization&Dropout浅析
一. Batch Normalization 对于深度神经网络,训练起来有时很难拟合,可以使用更先进的优化算法,例如:SGD+momentum.RMSProp.Adam等算法.另一种策略则是高改变网络 ...
- MYSQL,触发器,实现两个表共用ID不重复
前后台没有分开,为了区分前后台用户,所以分表,但是ID不能重复,因为关联了权限表. 这里实现后台用户表使用奇数ID 前台用户表使用偶数ID MYSQL 没有sequence SET @@auto_in ...
- 对于JAVA程序优化的一些想法,读书有感.治疗强迫症良药
在深入了解Java虚拟机里读到:在try{}块里面执行代码,比if(x!=null)效率要高,前提是被catch的几率很低的情况下. 但是 在Effective Java里读到:因为异常机制的设计初衷 ...
- SecureCRT + Tmux 分屏 高效开发
最近发现了SecureCRT的一些好玩的功能, 具体如下: 1. 发送消息到所有的终端 首先选中查看-->交互窗口 此时会看到下面出现一个输入窗口 然后, 右击选择"发送交互到所有标签 ...