POJ 2157 Evacuation Plan [最小费用最大流][消圈算法]
---恢复内容开始---
题意略。
这题在poj直接求最小费用会超时,但是题意也没说要求最优解。
根据线圈定理,如果一个跑完最费用流的残余网络中存在负权环,那么顺着这个负权环跑流量为1那么会得到更小的费用。
关键是坑在找环的起点。其实看了代码之后发现的确不难...
#include<stdio.h>
#include<queue>
#include<string.h>
#define MAXN 300
#define MAXM 30002*4
#define INF 10000000
using namespace std;
//起点编号必须最小,终点编号必须最大
bool vis[MAXN]; //spfa中记录是否在队列里边
int sum[],nnum[MAXN],zhong;
struct point{
int x,y,c;
}from[],to[];
struct edge{
edge *next,*op; //op是指向反向边
int t,c,v; //t下一个点编号,c容量,v权值
}ES[MAXM],*V[MAXN]; //ES边静态邻接表,V点的编号
int N,M,S,T,EC=-; //S源点最小,T汇点最大,EC当前边数
int demond[MAXN],sp[MAXN],prev[MAXN]; //spSPFA中记录距离,prev记录上一个点路径
edge *path[MAXN]; //与prev同步记录,记录到上一条边
int cal(point a,point b){
return max(a.x-b.x,b.x-a.x)+max(a.y-b.y,b.y-a.y)+;
}
void addedge(int a,int b,int v,int c,int cc){
edge e1={V[a],,b,c,v},e2={V[b],,a,cc,-v};
ES[++EC]=e1;V[a]=&ES[EC];
ES[++EC]=e2;V[b]=&ES[EC];
V[a]->op=V[b];V[b]->op=V[a];
}
bool SPFA(int st,int up){
int u,v;
memset(nnum,,sizeof(nnum));
memset(vis,,sizeof(vis));
for(u=;u<=up;u++){
sp[u]=INF;
}
queue<int>q;
q.push(st);
sp[st]=;
vis[st]=;
nnum[st]++;
while(!q.empty()){
u=q.front();
vis[u]=;
q.pop();
for(edge *k=V[u];k;k=k->next){
v=k->t;
if(k->c>&&sp[u]+k->v<sp[v]){
sp[v]=sp[u]+k->v;
prev[v]=u;
path[v]=k;
if(vis[v]==){
nnum[v]++;
if(nnum[v]>=up){
zhong=v;
return ;
}
vis[v]=;
q.push(v);
}
}
}
}
return ;
}
int argument(){
memset(vis,,sizeof(vis));
int i,cost=,flow=;
edge *e;
for(i=zhong;;i=prev[i]){
if(vis[i]==)vis[i]=;
else break;
}
zhong=i;
bool ok=;
memset(vis,,sizeof(vis));
for(i=zhong;;){
if(!vis[i])vis[i]=;
else break;
e=path[i];
e->c-=cost;e->op->c+=cost;
flow+=e->v*cost;
vis[i]=;
i=prev[i];
//if(i==zhong)break;
}
return flow;
}
int maxcostflow(int tans,int up){
int Flow=tans;
if(SPFA(,up)){
Flow+=argument();
}
return Flow;
}
void init(){
int n,m;
while(scanf("%d%d",&n,&m)!=EOF){
EC=-;
memset(sum,,sizeof(sum));
int tans=;
for(int i=;i<MAXN;i++){
V[i]=NULL;
}
for(int i=;i<n;i++){
scanf("%d%d%d",&from[i].x,&from[i].y,&from[i].c);
}
for(int i=;i<m;i++){
scanf("%d%d%d",&to[i].x,&to[i].y,&to[i].c);
}
for(int i=;i<n;i++){
for(int j=;j<m;j++){
int tmp;
scanf("%d",&tmp);
addedge(i+,j++n,cal(from[i],to[j]),INF,tmp);
tans+=cal(from[i],to[j])*tmp;
sum[j]+=tmp;
}
}
T=n+m+;
for(int i=;i<=m;i++){
addedge(i+n,T,,to[i-].c-sum[i-],sum[i-]);
}
int ans=maxcostflow(tans,T);
if(ans==tans)puts("OPTIMAL");
else{
puts("SUBOPTIMAL");
for(int i=;i<n;i++){
for(int j=;j<m;j++){
for(edge *k=V[j++n];k;k=k->next){
if(k->t==i+){
printf("%d",k->c);
if(j!=m-)printf(" ");
else printf("\n");
break;
}
}
}
}
}
}
}
int main(){
init();
return ;
}
---恢复内容结束---
POJ 2157 Evacuation Plan [最小费用最大流][消圈算法]的更多相关文章
- POJ 2175:Evacuation Plan(费用流消圈算法)***
http://poj.org/problem?id=2175 题意:有n个楼,m个防空洞,每个楼有一个坐标和一个人数B,每个防空洞有一个坐标和容纳量C,从楼到防空洞需要的时间是其曼哈顿距离+1,现在给 ...
- poj 2351 Farm Tour (最小费用最大流)
Farm Tour Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 17230 Accepted: 6647 Descri ...
- POJ 3680 Intervals 最小费用最大流(MCMF算法)
题意:给出 n ,k 表示接下来给你 n 段开区间,每段区间都有它的权值,问选出一些区间,使它的权值最大,并且在实轴上的每个点,不得超过 k次被覆盖. 思路:首先要理解建图思路,首先有一个基图,相邻点 ...
- POJ 2195 - Going Home - [最小费用最大流][MCMF模板]
题目链接:http://poj.org/problem?id=2195 Time Limit: 1000MS Memory Limit: 65536K Description On a grid ma ...
- poj 2135 Farm Tour 最小费用最大流建图跑最短路
题目链接 题意:无向图有N(N <= 1000)个节点,M(M <= 10000)条边:从节点1走到节点N再从N走回来,图中不能走同一条边,且图中可能出现重边,问最短距离之和为多少? 思路 ...
- POJ 3680: Intervals【最小费用最大流】
题目大意:你有N个开区间,每个区间有个重量wi,你要选择一些区间,使得满足:每个点被不超过K个区间覆盖的前提下,重量最大 思路:感觉是很好想的费用流,把每个区间首尾相连,费用为该区间的重量的相反数(由 ...
- poj2175费用流消圈算法
题意: 有n个建筑,每个建筑有ai个人,有m个避难所,每个避难所的容量是bi,ai到bi的费用是|x1-x2|+|y1-y2|+1,然后给你一个n*m的矩阵,表示当前方案,问当前避难方案是否 ...
- POJ 2135 Farm Tour [最小费用最大流]
题意: 有n个点和m条边,让你从1出发到n再从n回到1,不要求所有点都要经过,但是每条边只能走一次.边是无向边. 问最短的行走距离多少. 一开始看这题还没搞费用流,后来搞了搞再回来看,想了想建图不是很 ...
- poj 2195 Going Home(最小费用最大流)
题目:http://poj.org/problem?id=2195 有若干个人和若干个房子在一个给定网格中,每人走一个都要一定花费,每个房子只能容纳一人,现要求让所有人进入房子,且总花费最小. 构造一 ...
随机推荐
- 【转】HTML - embed 与 object 之争
在 HTML 里嵌入文本和图片之外的事物,就会用到嵌入标签,而嵌入标签在各浏览器之间的不统一,一直是让开发人员很头痛的问题.一切都要从嵌入 SUN 公司的 Applet Java 小程序开始. 当时, ...
- MySQLdb模块的安装
在WINDOWS环境连接利用python连接MYSQL数据库的时候,需要安装python工具包MySQL-python,但是直接用pip install安装会出错,错误信息如下: _mysql.c(4 ...
- mysql edit
表外键5个相关性: cascade,restrict,set null,no action,default show character set ; show collation like ' ...
- 打印Java main方法执行的命令参数代码
RuntimeMXBean bean = ManagementFactory.getRuntimeMXBean(); List<String> jvmArgs = bean.getInpu ...
- [mysql] mysql 5.6.27 innodb 相关参数
mysql> show variables like '%innodb%';+------------------------------------------+--------------- ...
- Markdown Example
An h1 header Paragraphs are separated by a blank line. 2nd paragraph. Italic, bold, and monospace. I ...
- android 插件化开发 开源项目列表
开源的插件化框架 Qihoo360/DroidPlugin CtripMobile/DynamicAPK mmin18/AndroidDynamicLoader singwhatiwanna/dyna ...
- Linux下编译OpenSSL
编译环境 操作系统: Red Hat Enterprise Linux Server release 5.4 64-bit 编译工具: gcc (GCC) 4.1.2 20080704 (Red Ha ...
- [rootfs]Yaffs2
1. busybox: sudo apt-get install busybox(v1.21.1) 2. mkyaffs2image: http://www.aleph1.co.uk/gitweb/? ...
- VS 使用Sql Server 数据库增删改查
/// <summary> /// 执行查询语句,返回DataSet /// </summary> /// <param name="SQLString&quo ...