POJ-2159 最小费用最大流
自己写的第一道费用流,图建好一波板子AC。不过还是有几个地方有点迷。
先来看看思想:题意是平面图上每个点有若干个人和若干个房子,保证人的数量等于房子的数量。每个格子上可以同时容纳所有人,也允许一个人在一个房子的格子上但不进入房子。求所有人都找到一个房子住所走的距离的和的最小值。
典型的最小费用最大流问题,用0点作为源点,给人编号,然后从源点引入一条边连接人,容量为1,费用为0,给房子编号,每个人到每个房子连一条边,容量为1,费用为他们的曼哈顿距离。然后再设置一个点作为汇点,所有的房子都与汇点连一条边,容量为1,费用为0.求一遍最小费用流就行了。
const int N=1e5+10;
struct Edge
{
int to,next,cap,flow,cost;
} e[N*10];
int head[N],tot;
int pre[N],dis[N];
int vis[N],Vis[201][201];
int n,m,tn;
char s[201][201];
void init(int num)
{
tot=0;//边的数量
tn=num;
memset(head,-1,sizeof(head));
}
void add(int u,int v,int cap,int cost)
{
e[tot].to=v,e[tot].cap=cap,e[tot].cost=cost,e[tot].flow=0;
e[tot].next=head[u];
head[u]=tot++;
e[tot].to=u,e[tot].cap=0,e[tot].cost=-cost,e[tot].flow=0;
e[tot].next=head[v];
head[v]=tot++;
}
bool spfa(int s,int t)
{
queue<int>q;
for(int i=0; i<=tn; i++)
{
dis[i]=INF;
vis[i]=0;
pre[i]=-1;
}
dis[s]=0;
vis[s]=1;
q.push(s);
while(!q.empty())
{
int u=q.front();
q.pop();
vis[u]=0;
for(int i=head[u]; i+1; i=e[i].next)
{
int v=e[i].to;
if(e[i].cap>e[i].flow&&dis[v]>dis[u]+e[i].cost)
{
dis[v]=dis[u]+e[i].cost;
pre[v]=i;
if(!vis[v])
{
vis[v]=1;
q.push(v);
}
}
}
}
if(pre[t]==-1) return false;
return true;
}
int mincost_maxflow(int s,int t)
{
int cost=0;
while(spfa(s,t))
{
int Min=INF;
for(int i=pre[t]; i!=-1; i=pre[e[i^1].to])
if(Min>e[i].cap-e[i].flow)
Min=e[i].cap-e[i].flow;
for(int i=pre[t]; i!=-1; i=pre[e[i^1].to])
{
e[i].flow+=Min;
e[i^1].flow-=Min;
cost+=e[i].cost*Min;
}
}
return cost;
}
struct node
{
int x,y;
};
vector<node>M,H;
int main()
{
while(~scanf("%d%d",&n,&m))
{
if(n==0&&m==0) return 0;
M.clear();
H.clear();
int num=1,tmp=1;
for(int i=0; i<n; i++)
{
scanf("%s",s[i]);
for(int j=0; j<m; j++) if(s[i][j]!='.') tmp++;
}
init(tmp);
for(int i=0; i<n; i++)
for(int j=0; j<m; j++)
if(s[i][j]!='.')
{
if(s[i][j]=='m')
{
add(0,num,1,0);//0为源点;
M.push_back(node {i,j});
}
else
{
add(num,tmp,1,0);//m+h+1为汇点
H.push_back(node {i,j});
}
Vis[i][j]=num++;
}
for(int i=0; i<M.size(); i++)
{
node tmpM=M[i];
for(int j=0; j<H.size(); j++)
{
node tmpH=H[j];
int diss=abs(tmpM.x-tmpH.x)+abs(tmpM.y-tmpH.y);
add(Vis[tmpM.x][tmpM.y],Vis[tmpH.x][tmpH.y],1,diss);
}
}
printf("%d\n",mincost_maxflow(0,tmp));
}
return 0;
}
POJ-2159 最小费用最大流的更多相关文章
- poj 3422(最小费用最大流)
题目链接:http://poj.org/problem?id=3422 思路:求从起点到终点走k次获得的最大值,最小费用最大流的应用:将点权转化为边权,需要拆点,边容量为1,费用为该点的点权,表示该点 ...
- POJ 2516 最小费用最大流
每一种货物都是独立的,分成k次最小费用最大流即可! 1: /** 2: 因为e ==0 所以 pe[v] pe[v]^1 是两条相对应的边 3: E[pe[v]].c -= aug; E[pe[v]^ ...
- POJ - 2195 最小费用最大流
题意:每个人到每个房子一一对应,费用为曼哈顿距离,求最小的费用 题解:单源点汇点最小费用最大流,每个人和房子对于建边 #include<map> #include<set> # ...
- poj 2195 最小费用最大流模板
/*Source Code Problem: 2195 User: HEU_daoguang Memory: 1172K Time: 94MS Language: G++ Result: Accept ...
- POJ 2135 最小费用最大流 入门题
Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19207 Accepted: 7441 Descri ...
- poj 2135最小费用最大流
最小费用最大流问题是经济学和管理学中的一类典型问题.在一个网络中每段路径都有"容量"和"费用"两个限制的条件下,此类问题的研究试图寻找出:流量从A到B,如何选择 ...
- poj 3680(最小费用最大流)
题目链接:http://poj.org/problem?id=3680 思路:因为N<=200,而区间范围为[1,100000],因此需要离散化,去重,然后就是建图了相连两点连边,容量为k,费用 ...
- POJ 2315 最小费用最大流
从1走到N然后从N走回来的最短路程是多少? 转换为费用流来建模. 1: /** 2: 因为e ==0 所以 pe[v] pe[v]^1 是两条相对应的边 3: E[pe[v]].c -= aug; E ...
- POJ 2135 最小费用最大流
题目链接 Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 18961 Accepted: 7326 D ...
- POJ 2195:Going Home(最小费用最大流)
http://poj.org/problem?id=2195 题意:有一个地图里面有N个人和N个家,每走一格的花费是1,问让这N个人分别到这N个家的最小花费是多少. 思路:通过这个题目学了最小费用最大 ...
随机推荐
- Linux一键脚本合集vps
首先,想说说一键脚本流行的原因何在? 众所周知的是,Linux 是占据大半壁江山的服务器系统,但在桌面上的占有率可就远不是那么回事儿了,使用和熟悉 Linux 的人远没有 Windows 多,但又因为 ...
- OO第13-14次作业总结
目录 面向对象第13-14次作业总结博客 1.设计分析 2.架构总结.测试 3.课程收获和建议 面向对象第13-14次作业总结博客 1.设计分析 这个单元是我做的最差的一个单元.总工程量超过2000行 ...
- dojo中类的继承
类似于c# java等后台语言,在基于类的面向对象编程中,通常需要在子类中扩展某些父类的方法,这时可以在子类的方法中,先调用从父类继承的方法,然后再执行子类自定义的操作.凡是使用declare创建的类 ...
- java 字符串中是否有数字
http://www.cnblogs.com/zhangj95/p/4198822.html http://www.cnblogs.com/sunzn/archive/2013/07/12/31865 ...
- tensorfow install error
http://stackoverflow.com/questions/33655731/error-while-importing-tensorflow-in-python2-7-in-ubuntu- ...
- C# 创建子目录
运用DirectoryInfo类创建子目录是非常容易的,你只要调用其中CreateSubdirectory()方法即可,演示代码如下. DirectoryInfo dir = new Director ...
- C#事件与接口编程实例
很多初学c#的朋友对于事件与接口感到迷惑不解,不明白它们之间的关系,下面我就用实例来简单的分析讲解一下. 事件,用event修饰符来代表一个事件,我们要创建一个C#事件必须按以下顺序来扫行: 1,创建 ...
- SAP销售订单屏幕增强行项目屏幕增强
1.在vbap表中 append一个自定义结构,如下图: 2.TCODE:SE80 程序名:SAPMV45A 屏幕:8459 如图: 3.标记增强的屏幕字段 4.屏幕增强的位置 *& ...
- 用dom1来实现,根据光标移动自动给表单加上背景色,光标移开自动去除背景色
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- ceph 性能
mysql在以下设备备份耗时,供大家参考: 备份文件大小 sata用时 ceph用时 nas挂载sata盘用时 7G 1分钟 15G 2分钟 21分钟 47G 8分钟 82分钟 274 ...