题意抽象出来就是给了一个费用流的残存网络,判断该方案是不是最优方案,如果不是,还要求给出一个更优方案。

在给定残存网络上检查是否存在负环即可判断是否最优。

沿负环增广一轮即可得到更优方案。

考虑到制作模板的需要,我用前向星建图做的。此题用邻接矩阵应该更方便。

#include<queue>
#include<cstdio>
#include<algorithm>
#include<cstring>
#define rep(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int maxn=;
const int MAXN=;
int nbuild,nshelt;
int x[maxn],y[maxn];
int tot,S,T;
int pointer[maxn];
int toT[MAXN];
int flo[maxn][maxn];
const int INF=0x3f3f3f;
int dis[maxn],pre[maxn],in[maxn],use[maxn];
bool vis[maxn];
bool visit[maxn];
struct Edge
{
int to,next,cap,f,w;
Edge() {};
Edge(int b,int c,int nxt,int flow,int weight) {to=b,cap=c,next=nxt,f=flow,w=weight;}
}edge[MAXN];
inline void addedge(int a,int b,int c,int flow,int w1)
{
edge[tot]=Edge(b,c,pointer[a],flow,w1);
pointer[a]=tot++;
edge[tot]=Edge(a,,pointer[b],-flow,-w1);
pointer[b]=tot++;
}
inline int cost(int a,int b)
{
return abs(x[a]-x[b])+abs(y[a]-y[b])+;
}
void init()
{
int t;
memset(pointer,-,sizeof(pointer));
memset(visit,,sizeof(vis));
tot=;
S=;T=nbuild+nshelt+;
rep(i,,nbuild)
{
scanf("%d%d%d",&x[i],&y[i],&t);
addedge(S,i,t,t,);
}
rep(i,nbuild+,nbuild+nshelt)
{
scanf("%d%d%d",&x[i],&y[i],&t);
toT[i]=tot;
addedge(i,T,t,,); }
int aedge;
rep(i,,nbuild)
{
rep(j,,nshelt)
{
scanf("%d",&t);
addedge(i,nbuild+j,INF,t,cost(i,nbuild+j));
// printf("%d-->%d w=%d\n",i,nbuild+j,cost(i,nbuild+j));
flo[i][j]=t;
aedge=toT[nbuild+j];
edge[aedge].f+=t;
edge[aedge^].f-=t;
}
} }
int ve=;
void clearcircle()
{
memset(use,,sizeof(use));
int enter;
while()
{
use[ve]++;
// printf("ve=%d\n",ve);
if(use[ve]>=) {enter=ve;break;}
ve=edge[pre[ve]^].to;
}
int sub;
do
{
sub=edge[pre[ve]^].to;
edge[pre[ve]].f+=;
edge[pre[ve]^].f-=;
ve=sub;
}while(ve!=enter);
rep(i,,nbuild)
{
for(int j=pointer[i];j!=-;j=edge[j].next)
{
int v=edge[j].to; if(v>nbuild) flo[i][v-nbuild]=edge[j].f;
}
} }
int searchcircle(int s)
{
queue<int> q;
rep(i,S,T)
{
dis[i]=INF;
vis[i]=;
pre[i]=-;
in[i]=;
}
q.push(s);
vis[s]=;
in[s]++;dis[s]=;
while(!q.empty())
{
int u=q.front();
q.pop();
vis[u]=;
// printf("%d dis[u]=%d\n",u,dis[u]);
for(int j=pointer[u];j!=-;j=edge[j].next)
{
int v=edge[j].to;
// printf("%d-->%d vis[v]=%d dis[v]=%d ds[u]=%d edge[j].w=%d\n",u,v,vis[v],dis[v],dis[u],edge[j].w);
if(edge[j].cap-edge[j].f&&dis[v]>dis[u]+edge[j].w)
{
dis[v]=dis[u]+edge[j].w;
pre[v]=j;
// printf("%d-->%d vis[v]=%d dis[v]=%d ds[u]=%d edge[j].w=%d\n",u,v,vis[v],dis[v],dis[u],edge[j].w);
if(!vis[v])
{
vis[v]=;
visit[v]=;
in[v]++;
q.push(v);
if(in[v]>=T)
{
ve=v;
return ;
}
}
}
}
}
return ; }
int main()
{
// freopen("in.txt","r",stdin);
while(scanf("%d",&nbuild)==)
{
scanf("%d",&nshelt);
init();
/*rep(i,1,nbuild)
{
rep(j,1,nshelt) printf("%d ",flo[i][j]);
printf("\n");
}
rep(i,1,nbuild)
{
rep(j,1,nshelt) printf("%d ",cost(i,nbuild+j));
printf("\n");
}*/
int flag=;
ve=;
rep(i,S,T)
{
if(visit[i]) continue;
ve=searchcircle(i);
if(ve)
{
flag=;
printf("SUBOPTIMAL\n");
clearcircle();
rep(i,,nbuild)
{
rep(j,,nshelt) printf("%d ",flo[i][j]);
printf("\n");
}
break;
}
}
if(!flag) printf("OPTIMAL\n"); }
return ;
}

poj 2175 费用流消圈的更多相关文章

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

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

  2. poj2175费用流消圈算法

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

  3. POJ.2175.Evacuation Plan(消圈)

    POJ \(Description\) \(n\)个建筑物,每个建筑物里有\(a_i\)个人:\(m\)个避难所,每个避难所可以容纳\(b_i\)个人. 给出每个建筑物及避难所的坐标,任意两点间的距离 ...

  4. POJ 2175 spfa费用流消圈

    题意:给出n栋房子位置和每栋房子里面的人数,m个避难所位置和每个避难所可容纳人数.然后给出一个方案,判断该方案是否最优,如果不是求出一个更优的方案. 思路:很容易想到用最小费用流求出最优时间,在与原方 ...

  5. POJ 2157 Evacuation Plan [最小费用最大流][消圈算法]

    ---恢复内容开始--- 题意略. 这题在poj直接求最小费用会超时,但是题意也没说要求最优解. 根据线圈定理,如果一个跑完最费用流的残余网络中存在负权环,那么顺着这个负权环跑流量为1那么会得到更小的 ...

  6. poj 2516 (费用流)

    题意:有N个供应商,M个店主,K种物品.每个供应商对每种物品的的供应量已知,每个店主对每种物品的需求量的已知,从不同的供应商运送不同的货物到不同的店主手上需要不同的花费,又已知从供应商m送第k种货物的 ...

  7. POJ 2175 Evacuation Plan 费用流 负圈定理

    题目给了一个满足最大流的残量网络,判断是否费用最小. 如果残量网络中存在费用负圈,那么不是最优,在这个圈上增广,增广1的流量就行了. 1.SPFA中某个点入队超过n次,说明存在负环,但是这个点不一定在 ...

  8. 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 ...

  9. poj 3422 (费用流)

    从左上角到有下角k次能获得的最大值. 跟hdu 2686一样的题目,这题一个点可以重复走,只能得到一次值. #include<stdio.h> #include<string.h&g ...

随机推荐

  1. MapReduce分布式算法

    一.MapReduce 有一种特殊的并行算法,就是分布式算法.在并行算法只需要两个到四个内核时,完全可以在笔记本电脑上运行,但是如果需要数百个内核,这种情况下,可让算法在多台计算机上运行. 假设你有一 ...

  2. Linux -- 基于zookeeper的java api(二)

    Linux -- 基于zookeeper的java api(二) 写一个关于基于集群的zookeeper的自定义实现HA 基于客户端和监控器:使用监控的方法查看每个注册过的节点的状态来做出操作. Wa ...

  3. JNDI学习总结(一)——JNDI数据源的配置(转)

    一.数据源的由来 在Java开发中,使用JDBC操作数据库的四个步骤如下:   ①加载数据库驱动程序(Class.forName("数据库驱动类");)   ②连接数据库(Conn ...

  4. npm使用过程中出现的错误

    1.安装npm install axios -S报错install "npm ERR! Error: EPERM: operation not permitted" error 经 ...

  5. 最新jquery+easyui_api培训文档

    目  录 1 Accordion(可折叠标签) 2 1.1 实例 2 1.2 参数 3 2 DateBox(日期框) 4 2.1 实例 4 2.2 参数 6 2.3 事件 6 2.4 方法 6 3 C ...

  6. JAVA设计模式(二)工厂模式

    在面向对象编程中, 最通常的方法是一个new操作符产生一个对象实例,new操作符就是用来构造对象实例的.但是在一些情况下, new操作符直接生成对象会带来一些问题.举例来说, 许多类型对象的创造需要一 ...

  7. 归并排序(Merging Sort)

    body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...

  8. Centos7下mysql5.7.22主从配置

    一:服务器配置 1.1修改root密码: ALTER USER 'root'@'localhost' IDENTIFIED BY 'xxxx'; 1.2添加远程登陆用户: GRANT ALL PRIV ...

  9. redis 解析配置文件

    在redis安装文件夹里面有redis.conf,查看配置. 一:基础配置介绍 1.units(单位) --这里可以看到 1k和1kb是不一样的,  units 这里单位是不区分大小写的,are al ...

  10. C#正则表达式MatchCollection类

    认识MatchCollection 类 表示通过以迭代方式将正则表达式模式应用于输入字符串所找到的成功匹配的集合. 命名空间:  System.Text.RegularExpressions 属性:C ...