---恢复内容开始---

题意略。

这题在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 [最小费用最大流][消圈算法]的更多相关文章

  1. POJ 2175:Evacuation Plan(费用流消圈算法)***

    http://poj.org/problem?id=2175 题意:有n个楼,m个防空洞,每个楼有一个坐标和一个人数B,每个防空洞有一个坐标和容纳量C,从楼到防空洞需要的时间是其曼哈顿距离+1,现在给 ...

  2. poj 2351 Farm Tour (最小费用最大流)

    Farm Tour Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 17230   Accepted: 6647 Descri ...

  3. POJ 3680 Intervals 最小费用最大流(MCMF算法)

    题意:给出 n ,k 表示接下来给你 n 段开区间,每段区间都有它的权值,问选出一些区间,使它的权值最大,并且在实轴上的每个点,不得超过 k次被覆盖. 思路:首先要理解建图思路,首先有一个基图,相邻点 ...

  4. POJ 2195 - Going Home - [最小费用最大流][MCMF模板]

    题目链接:http://poj.org/problem?id=2195 Time Limit: 1000MS Memory Limit: 65536K Description On a grid ma ...

  5. poj 2135 Farm Tour 最小费用最大流建图跑最短路

    题目链接 题意:无向图有N(N <= 1000)个节点,M(M <= 10000)条边:从节点1走到节点N再从N走回来,图中不能走同一条边,且图中可能出现重边,问最短距离之和为多少? 思路 ...

  6. POJ 3680: Intervals【最小费用最大流】

    题目大意:你有N个开区间,每个区间有个重量wi,你要选择一些区间,使得满足:每个点被不超过K个区间覆盖的前提下,重量最大 思路:感觉是很好想的费用流,把每个区间首尾相连,费用为该区间的重量的相反数(由 ...

  7. poj2175费用流消圈算法

    题意:      有n个建筑,每个建筑有ai个人,有m个避难所,每个避难所的容量是bi,ai到bi的费用是|x1-x2|+|y1-y2|+1,然后给你一个n*m的矩阵,表示当前方案,问当前避难方案是否 ...

  8. POJ 2135 Farm Tour [最小费用最大流]

    题意: 有n个点和m条边,让你从1出发到n再从n回到1,不要求所有点都要经过,但是每条边只能走一次.边是无向边. 问最短的行走距离多少. 一开始看这题还没搞费用流,后来搞了搞再回来看,想了想建图不是很 ...

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

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

随机推荐

  1. python sys.argv[]

    sys.argv[]是用来获取命令行参数的,是一个由该脚本自身路径和其它输入的参数组成的List.sys.argv[0]表示代码本身文件路径. 这里,当我们执行python using_sys.py ...

  2. 运用cookie登陆人人网爬取数据

    浏览器访问WEB服务器的过程 在用户访问网页时,不论是通过URL输入域名或IP,还是点击链接,浏览器向WEB服务器发出了一个HTTP请求(Http Request),WEB服务器接收到客户端浏览器的请 ...

  3. Knockout应用开发指南(完整版) 目录索引

    http://learn.knockoutjs.com/  所有示例和代码都在在上面直接运行预览 注意:因为它用了google的cdn加速,所要要用代_理+_翻_墙才能正常加载 使用Knockout有 ...

  4. [转]oracle for update和for update nowait的区别

    1概念小结:(针对以下引用区域内容) 1.1 普通select语句不加锁. 1.2 for update和for update nowait都试图将符合条件的数据加上行级锁.用于排斥其他针对这个表的写 ...

  5. javascript 定义正则表达式

    js中定义正则表达式有两种,使用RegExp和使用字面量. 使用字面量定义时需要注意:必须以/开始,以/结束,就像定义字符串一样("test"). 但是,js的正则表达式可以通过指 ...

  6. android外包公司——最新案例铁血军事手机客户端(IOS & Android)

    <铁血军事>Android手机客户端由铁血网开发和运营,为网友提供铁血论坛和铁血读书两大产品.使用Android手机客户端,您不仅可以阅读到最新军事资讯,随时参与精彩话题讨论,还可以在线阅 ...

  7. VS2013 启动时遇到空白窗口

    安装了VS2013 ,然后打开,遇到空白窗口,等了10几秒无果. 我先是修复了VS2013,然后再打开VS2013,则显示登录微软账号的窗口.修复完2013时正好18:08,可能是下班了可以上网,才正 ...

  8. Saltstack系列5:Saltstack之pillar组件

    pillar组件 pillar也是Saltstack最重要的组件之一,其作用是定义与被控主机相关的任何数据,定义好的数据可以被其他组件使用,如模板.state.API等. 在pillar中定义的数据与 ...

  9. SVN Server配置详解 及备份

    SVN简介和工作原理 subversion(简称svn)是近几年崛起的版本管理软件,是cvs的接班人,目前绝大多数开源软件都使用svn作为代码版本管理软件.Subversion支持linux和wind ...

  10. SPOJ Query on a tree 树链剖分 水题

    You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...