/*
做网络流的题建图真的是太重要了!
本题是将人所在的位置和房子所在的位置建立边的联系,其中man到house这一条边的流量为 1, 费用为两者的距离
而方向边的流量为 0, 费用为正向边的相反数(也就是沿着反向边进行增广时,费用要减少,更改先前错误的选择)
最后增加一个源点和一个汇点完毕(源点映射到man, house映射到汇点, 费用为0, 流量为1)
    
*/
#include<iostream>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<queue>
#define Max 0x3f3f3f3f
#define N 205
using namespace std; class node
{
public:
int x, y;
}; node xyM[N];
node xyH[N];
int cost[N][N], cap[N][N];
int cntM, cntH;
int pre[N*], dist[N*], vis[N*], n, m; void addE(int i, int j, int ct, int cp)
{
cost[i][j]=ct;
cap[i][j]=cp;
cost[j][i]=-ct;
//cap[j][i]=0;
} int s, t, minCost; void buildMap()
{
int i, j;
memset(cap, , sizeof(cap));
s=; t=cntM+cntH+;
for(i=; i<cntM; ++i)
addE(, i+, , );
for(i=; i<cntH; ++i)
addE(cntM+i+, t, , );
for(i=; i<cntM; ++i)
for(j=; j<cntH; ++j)
addE(i+, cntM+j+, abs(xyM[i].x-xyH[j].x)+abs(xyM[i].y-xyH[j].y), );
} queue<int>q; int spfa()
{
int u, v;
memset(dist, 0x3f, sizeof(dist));
dist[]=;
q.push();
vis[]=;
while(!q.empty())
{
u=q.front();
q.pop();
vis[u]=;
for(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(!vis[v])
{
vis[v]=;
q.push(v);
}
}
}
if(dist[t]==Max)
return ;
return ;
} void updateEdge()
{
int u, minFlow=Max;
for(u=t; u!=s; u=pre[u])//通过最短路径寻找这条路径上的最小流量
if(cap[pre[u]][u]<minFlow)
minFlow=cap[pre[u]][u];
for(u=t; u!=s; u=pre[u])
{
cap[pre[u]][u]-=minFlow;
cap[u][pre[u]]+=minFlow;
minCost+=cost[pre[u]][u];
}
} int main()
{
int i, j;
char c;
while(scanf("%d%d", &n, &m) && (n||m))
{
cntM=cntH=;
minCost=;
for(i=; i<=n; ++i)
{
getchar();
for(j=; j<=m; ++j)
{
scanf("%c", &c);
if(c=='m')
{
xyM[cntM].x=i;
xyM[cntM++].y=j;
}
else if(c=='H')
{
xyH[cntH].x=i;
xyH[cntH++].y=j;
}
}
}
buildMap();
while(spfa())
updateEdge();
printf("%d\n", minCost);
}
return ;
}

//邻接表

 #include<iostream>
#include<queue>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define INF 0x3f3f3f3f
#define N 1000005
using namespace std; int cntH, cntM; struct node{
int x, y;
}; struct EDGE{
int u, v, cap, cost, nt;
};
EDGE edge[N]; queue<int>q;
node man[], house[];
int first[];
int dist[];
int pre[], flow[], vis[];
int cnt, t;
int minCost;
int n, m; void addEdge(int u, int v, int cap, int cost){
edge[cnt].u=u;
edge[cnt].v=v;
edge[cnt].cap=cap;
edge[cnt].nt=first[u];
edge[cnt].cost=cost;
first[u]=cnt++; edge[cnt].u=v;
edge[cnt].v=u;
edge[cnt].cap=;
edge[cnt].nt=first[v];
edge[cnt].cost=-cost;
first[v]=cnt++;
} void buildMap(){
memset(first, -, sizeof(first));
t=cntH+cntM+;
for(int i=; i<=cntM; ++i)
for(int j=; j<=cntH; ++j)
addEdge(i, cntM+j, , abs(man[i].x-house[j].x) + abs(man[i].y-house[j].y));
for(int i=; i<=cntM; ++i)
addEdge(, i, , );
for(int i=; i<=cntH; ++i)
addEdge(cntM+i, t, , );
} bool MCMF(){
memset(dist, 0x3f, sizeof(dist));
memset(vis, , sizeof(vis));
q.push();
flow[]=INF;
dist[]=;
vis[]=;
while(!q.empty()){
int u=q.front(); q.pop();
vis[u]=;
for(int e=first[u]; ~e; e=edge[e].nt){
int v=edge[e].v, cap=edge[e].cap, cost=edge[e].cost;
if(cap> && dist[v]>dist[u]+cost){
dist[v]=dist[u]+cost;
flow[v]=min(flow[u], cap);
pre[v]=e;
if(!vis[v]){
vis[v]=;
q.push(v);
}
}
}
}
if(dist[t]==INF) return false;
minCost+=dist[t];
int x=t;
while(x!=){
edge[pre[x]].cap-=flow[t];
edge[pre[x]^].cap+=flow[t];
x=edge[pre[x]].u;
}
return true;
} int main(){
while(scanf("%d%d", &n, &m) && (n||m)){
cnt=cntH=cntM=;
for(int i=; i<=n; ++i){
getchar();
for(int j=; j<=m; ++j){
char ch;
scanf("%c", &ch);
if(ch=='m'){
man[++cntM].x=i;
man[cntM].y=j;
}
else if(ch=='H'){
house[++cntH].x=i;
house[cntH].y=j;
}
}
}
buildMap();
minCost=;
while(MCMF());
printf("%d\n", minCost);
}
return ;
}

poj 2195 Going Home的更多相关文章

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

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

  2. poj 2195 二分图带权匹配+最小费用最大流

    题意:有一个矩阵,某些格有人,某些格有房子,每个人可以上下左右移动,问给每个人进一个房子,所有人需要走的距离之和最小是多少. 貌似以前见过很多这样类似的题,都不会,现在知道是用KM算法做了 KM算法目 ...

  3. POJ 2195 Going Home / HDU 1533(最小费用最大流模板)

    题目大意: 有一个最大是100 * 100 的网格图,上面有 s 个 房子和人,人每移动一个格子花费1的代价,求最小代价让所有的人都进入一个房子.每个房子只能进入一个人. 算法讨论: 注意是KM 和 ...

  4. POJ 2195 Going Home (带权二分图匹配)

    POJ 2195 Going Home (带权二分图匹配) Description On a grid map there are n little men and n houses. In each ...

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

    poj 2195 Going Home Description On a grid map there are n little men and n houses. In each unit time ...

  6. 【POJ 2195】 Going Home(KM算法求最小权匹配)

    [POJ 2195] Going Home(KM算法求最小权匹配) Going Home Time Limit: 1000MS   Memory Limit: 65536K Total Submiss ...

  7. POJ 2195:Going Home(最小费用最大流)

    http://poj.org/problem?id=2195 题意:有一个地图里面有N个人和N个家,每走一格的花费是1,问让这N个人分别到这N个家的最小花费是多少. 思路:通过这个题目学了最小费用最大 ...

  8. poj 2195 KM算法

    题目链接:http://poj.org/problem?id=2195 KM算法模板~ 代码如下: #include "stdio.h" #include "string ...

  9. Poj(2195),最小费用流,SPFA

    题目链接:http://poj.org/problem?id=2195 Going Home Time Limit: 1000MS   Memory Limit: 65536K Total Submi ...

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

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

随机推荐

  1. CCNA第二章TCP/IP简介考试要点学习笔记

    1.描述网络是如何工作的     DoD过程/应用层 -- OSI应用.表示和会话层(定义了结点到结点的应用通信协议以及对用户界面规范的控制): DoD主机到主机层 -- OSI传输层(保证了数据包的 ...

  2. jsti 和EL用法注意点

    今天使用stl 结合El做jsp页面展现,出现了个问题,怎么调也调不好,最后将jstl的源码拿来跟踪调了一下才明白其中的道理. 在使用jstl tag <c:forEach var=" ...

  3. linux sort,uniq,cut,wc命令详解

    linux sort,uniq,cut,wc命令详解 sort sort 命令对 File 参数指定的文件中的行排序,并将结果写到标准输出.如果 File 参数指定多个文件,那么 sort 命令将这些 ...

  4. ios 随记录

    1. 设置全局的就不写了.当单个VC需要的时候.一般是这样子. /* 设置StatusBar的样式,UIStatusBarStyleDefault与UIStatusBarStyleLightConte ...

  5. NYOJ 536 开心的mdd(DP)

    开心的mdd 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 himdd有一天闲着无聊,随手拿了一本书,随手翻到一页,上面描述了一个神奇的问题,貌似是一个和矩阵有关的 ...

  6. C#基于Socket的简单聊天室实践

    序:实现一个基于Socket的简易的聊天室,实现的思路如下: 程序的结构:多个客户端+一个服务端,客户端都是向服务端发送消息,然后服务端转发给所有的客户端,这样形成一个简单的聊天室功能. 实现的细节: ...

  7. 配置WCF同时支持WSDL和REST,swaggerwcf生成文档

    配置WCF同时支持WSDL和REST,SwaggerWCF生成文档 VS创建一个WCF工程,通过NuGet添加SwaggerWcf 创建完成后通过 程序包管理控制台 pm>Install-Pac ...

  8. javascript 设计模式-----模块模式

    在一些大的项目中经常使用到模块,在这里,我们将了解一下什么是模块模式.模块模式最简单的方法大家一定会用过,如下所示: var a = { b : 1, c : 2 } 这样一个对象的直接量其实就已经是 ...

  9. Linux 网络编程(UDP)

    客户端代码 #include<stdio.h> #include<stdlib.h> #include<string.h> #include<sys/sock ...

  10. 【C语言学习】《C Primer Plus》第12章 存储类、链接和内存管理

    学习总结 1.作用域可分为代码块作用域.函数原型作用域或者文件作用域. 代码块作用域例子: { for(int i=0;i<10;i++){  //C99允许 …  //i的作用域 } ... ...