POJ 2396  Budget

题意简述:给定矩阵(每个元素都是非负整数)各行各列的和,并且限制其中的某些元素,给出一个可行解,特殊评测。矩阵规模小于200*20。

网络流的模型是显而易见的,不过对于这道题,我们要添加两次源和汇。

第一次添加s连接每一行,t连接每一列,容量上下线都是这行或这列的和。

第二次对每条有容量限制的边(u,v)添加 (ss,v)和( u,tt)容量均为( u,v)的下限。

第三次添加(t,s)容量无穷。

对(ss,tt)求最大流,若ss出发和进入tt的边均满流则有解,并且可以直接输出解。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<vector>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int maxm=200+50,INF=1e+8;
int k,ans[maxm][maxm],cap[maxm][maxm],flow[maxm][maxm],low[maxm][maxm],up[maxm][maxm],n,m;
vector<int> next[maxm]; int d[maxm],fa[maxm],cur[maxm];
bool vis[maxm];
bool bfs(int s,int t)
{
memset(vis,0,sizeof(vis));
memset(d,0,sizeof(d));
queue<int>q;
vis[s]=true;q.push(s);
while(!q.empty())
{
int np=q.front();q.pop();
for(int i=0;i<next[np].size();i++)
{
int ne=next[np][i];
if(cap[np][ne]<=flow[np][ne]||vis[ne])continue;
q.push(ne);
vis[ne]=true;
d[ne]=d[np]+1;
}
}
return vis[t];
} int dfs(int now,int t,int flo)
{
if(now==t||flo==0)return flo;
int floww=0;
for(int i=cur[now];i<next[now].size();i++)
{
int np=next[now][i];
if(d[np]!=d[now]+1)continue;
if(cap[now][np]<=flow[now][np])continue;
int fn=dfs(np,t,min(flo,cap[now][np]-flow[now][np]));
flow[now][np]+=fn;
flow[np][now]-=fn;
floww+=fn;
flo-=fn;
if(flo==0)break;
cur[now]++;
}
return floww;
}
int dinic(int s,int t)
{
int ans=0;
while(bfs(s,t))
{
memset(cur,0,sizeof(cur));
ans+=dfs(s,t,INF);
}
return ans;
}
void find(int s,int t)
{
memset(d,0,sizeof(d));
d[s]=INF;
queue<int>q;
q.push(s);
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=0;i<next[u].size();i++)
{
int nextp=next[u][i];
if(d[nextp]!=0||(cap[u][nextp]-flow[u][nextp])<=0)continue;
d[nextp]=min(d[u],(cap[u][nextp]-flow[u][nextp]));
q.push(nextp);
fa[nextp]=u;
if(nextp==t)
{
while(!q.empty())q.pop();
return ;
}
}
}
}
int E_K(int s,int t)
{
int ans=0;
memset(fa,0,sizeof(fa));
while(true)
{
find(s,t);
ans+=d[t];
if(d[t]==0)break;
for(int i=t;i!=s;i=fa[i])
{
flow[fa[i]][i]+=d[t];
flow[i][fa[i]]-=d[t];
}
}
return ans;
} void addedge(int a,int b,vector<int> next[])
{
next[a].push_back(b);
next[b].push_back(a);
return ;
}
int main()
{
int N;
int s=249,t=s-1,ss=t-1,tt=ss-1;
scanf("%d",&N);
for(int ii=1;ii<=N;ii++)
{ for(int i=0;i<maxm;i++)
while(next[i].size()>0)next[i].pop_back();
memset(cap,0,sizeof(cap));memset(flow,0,sizeof(flow));
memset(low,0,sizeof(low));memset(up,0,sizeof(up));
scanf("%d%d",&m,&n);
for(int i=1;i<=m;i++)
{
int a;scanf("%d",&a);
cap[s][i]=0;cap[ss][i]+=a;cap[s][tt]+=a;
addedge(s,i,next);addedge(ss,i,next);addedge(i,tt,next);
for(int j=201;j<=200+n;j++)
{
addedge(i,j,next);
up[i][j]=a;
}
}
addedge(s,tt,next);
for(int i=201;i<=200+n;i++)
{
int a;scanf("%d",&a);
cap[i][t]=0;cap[ss][t]+=a;cap[i][tt]+=a;
addedge(i,t,next);addedge(i,tt,next);addedge(i,ss,next);
for(int j=1;j<=m;j++)
up[j][i]=min(up[j][i],a);
}
addedge(ss,t,next);
int k;
scanf("%d",&k);
for(int iii=0;iii<k;iii++)
{
int a,b,c;char t;
scanf("%d%d",&a,&b);t=getchar();t=getchar();scanf("%d",&c);
int ia,ib,ja,jb;
if(a==0){ia=1;ib=m;}
else {ia=ib=a;}
if(b==0){ja=201;jb=200+n;}
else{ja=jb=(200+b);}
for(int i=ia;i<=ib;i++)
for(int j=ja;j<=jb;j++)
{
if(t=='=')
{
up[i][j]=min(up[i][j],c);low[i][j]=max(low[i][j],c);
}
if(t=='>')
{
low[i][j]=max(low[i][j],c+1);
}
if(t=='<')
{
up[i][j]=min(up[i][j],c-1);
}
}
}
for(int i=1;i<=m;i++)
for(int j=201;j<=200+n;j++)
{
cap[i][j]=up[i][j]-low[i][j];
cap[ss][j]+=low[i][j];
cap[i][tt]+=low[i][j];
}
next[t].push_back(s);cap[t][s]=INF;
dinic(ss,tt);
bool impo=false;
for(int i=1;i<=m;i++)
{
if(flow[ss][i]!=cap[ss][i]){impo=true;break;}
if(flow[i][tt]!=cap[i][tt]){impo=true;break;}
}
for(int j=201;j<=200+n;j++)
{
if(flow[ss][j]!=cap[ss][j]){impo=true;break;}
if(flow[j][tt]!=cap[j][tt]){impo=true;break;}
}
if(flow[s][tt]!=cap[s][tt]){impo=true;}
if(flow[ss][t]!=cap[ss][t]){impo=true;}
if(impo){printf("IMPOSSIBLE\n\n");continue;}
for(int i=1;i<=m;i++)
{ for(int j=201;j<n+200;j++)
{
printf("%d ",flow[i][j]+low[i][j]);
}
printf("%d\n",flow[i][200+n]+low[i][200+n]);
}
printf("\n");
}
return 0;
}

  

                                    

POJ 2396 Budget 有上下界的网络流的更多相关文章

  1. POJ 2396 Budget ——有上下界的网络流

    给定矩阵的每行每列的和,和一些大于小于等于的限制.然后需要求出一组可行解. 上下界网络流. 大概的思想就是计算出每一个点他需要强行流入或者流出的量,然后建出超级源点和汇点,然后删除下界,就可以判断是否 ...

  2. poj 2396 Budget 边容量有上下界的最大流

    题意: 给一个矩阵的每行和及每列和,在给一些行列或点的限制条件.求一个满足的矩阵. 分析: 转化为有上下界的网络流,注意等于也是一种上下界关系,然后用dinic算法. 代码: //poj 2396 / ...

  3. 【POJ2396】Budget(上下界网络流)

    Description We are supposed to make a budget proposal for this multi-site competition. The budget pr ...

  4. ACM/ICPC 之 有流量上下界的网络流-Dinic(可做模板)(POJ2396)

    //有流量上下界的网络流 //Time:47Ms Memory:1788K #include<iostream> #include<cstring> #include<c ...

  5. SGU 194. Reactor Cooling(无源汇有上下界的网络流)

    时间限制:0.5s 空间限制:6M 题意: 显然就是求一个无源汇有上下界的网络流的可行流的问题 Solution: 没什么好说的,直接判定可行流,输出就好了 code /* 无汇源有上下界的网络流 * ...

  6. 【ZOJ2314】Reactor Cooling(有上下界的网络流)

    前言 话说有上下界的网络流好像全机房就我一个人会手动滑稽,当然这是不可能的 Solution 其实这道题目就是一道板子题,主要讲解一下怎么做无源无汇的上下界最大流: 算法步骤 1.将每条边转换成0~u ...

  7. ZOJ 2314 有上下界的网络流

    problemCode=2314">点击打开链接 题意:给定m条边和n个节点.每条边最少的流量和最多的流量.保证每一个节点的出入流量和相等,问能够形成吗,能够则输出每条边的流量 思路: ...

  8. poj_2396 有上下界的网络流

    题目大意 一个mxn的矩阵,给出矩阵中每一行的和sh[1,2...m]以及每一列的数字的和目sv[1,2...n],以及矩阵中的一些元素的范围限制,比如a[1][2] > 1, a[2][3] ...

  9. 【BZOJ2502】清理雪道 有上下界的网络流 最小流

    [BZOJ2502]清理雪道 Description        滑雪场坐落在FJ省西北部的若干座山上. 从空中鸟瞰,滑雪场可以看作一个有向无环图,每条弧代表一个斜坡(即雪道),弧的方向代表斜坡下降 ...

随机推荐

  1. IOS三种归档(NSKeyArchieve)的总结

    IOS三种归档(NSKeyArchieve)的总结 归档是一种IOS中常用来存储文件的一种方法,在面向对象的语言中,归档也就实际上可以将一切对象存储在文件中,以下是IOS开发中常见的三种文件归档方式, ...

  2. 分30条依次解析xml并插入数据库成功

    package xxx; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import or ...

  3. UVa 10382 - Watering Grass

    题目大意:有一条长为l,宽为w的草坪,在草坪上有n个洒水器,给出洒水器的位置和洒水半径,求能浇灌全部草坪范围的洒水器的最小个数. 经典贪心问题:区间覆盖.用计算几何对洒水器的覆盖范围简单处理一下即可得 ...

  4. UVa 10033 - Interpreter

    题目大意:模拟题,有一些寄存器和随机访问内存,给你一些指令以及它们代表的操作,模拟操作即可. #include <cstdio> #include <cstring> #def ...

  5. linux环境下Vim的配置

    原文链接:http://blog.chinaunix.net/uid-26826958-id-3272375.html  (本文转自此链接中的部分内容,但做了适当修改) 安装vim命令:sudo ap ...

  6. Spring新下载地址

    Spring官网改版后找了好久都没有找到直接下载Jar包的链接,下面汇总些网上提供的方法,亲测可用. 1.直接输入地址,改相应版本即可:http://repo.springsource.org/lib ...

  7. js面向对象oop编程

    理解对象 对象这个词如雷贯耳,同样出名的一句话:XXX语言中一切皆为对象! 对象究竟是什么?什么叫面向对象编程? 对象(object),台湾译作物件,是面向对象(Object Oriented)中的术 ...

  8. BNU Online Judge-34776-What does the fox say?

    题目链接 http://www.bnuoj.com/bnuoj/problem_show.php?pid=34776 题意: fox 的叫声 例如测试用例 输入 toot woof wa ow ow ...

  9. 第一个Servlet程序及分析

    第一个Servlet程序: package cc.openhome; import java.io.IOException; import java.io.PrintWriter; import ja ...

  10. JAVA轻量级文件监控

    原文地址:http://blog.csdn.net/three_man/article/details/31012903?utm_source=tuicool 介绍 本文主要介绍一种轻量级的文件监控方 ...