POJ2396_Budget
题意为给一个矩形数字阵,给出一些限制条件,包括每行和每列的和,还有一些位置的数值范围,求出满足情况的一个。
首先建图,源点->行和->列和->汇点,显然,行和列之间的边为那个数字的大小,只要我们能够找到一个满足大小条件的,且使的两边的和满流的流量方案就可以了。
由于存在下界(上界其实就是边的容量),根据图的特殊性,我们可以先在那边的相连的两条边都减去这个下界,这样就变成了一条只有上界的边了。
召唤代码君:
#include <iostream>
#include <cstring>
#include <cstdio>
#define maxn 1022
#define maxm 844442
typedef long long ll;
using namespace std; const ll inf=~0U>>;
ll to[maxm],next[maxm],c[maxm],first[maxn],edge;
ll fmin[maxn][maxn],fmax[maxn][maxn],sr[maxn],sc[maxn];
ll d[maxn],tag[maxn],TAG=;
ll Q[maxm],bot,top;
ll ans[maxn][maxn];
bool can[maxn];
ll n,m,s,t,T,R,sumr,sumc; void _init()
{
s=,t=n+m+,edge=-,sumr=sumc=;
for (ll i=s; i<=t; i++) first[i]=-;
for (ll i=; i<=n; i++) sr[i]=;
for (ll i=; i<=m; i++) sc[i]=;
for (ll i=; i<=n; i++)
for (ll j=; j<=m; j++) ans[i][j]=,fmin[i][j]=,fmax[i][j]=inf;
} void minsize(ll x,ll y,ll dn,ll up)
{
fmin[x][y]=max(fmin[x][y],dn);
fmax[x][y]=min(fmax[x][y],up);
} bool check()
{
for (ll i=; i<=n; i++)
for (ll j=; j<=m; j++)
{
if (fmax[i][j]<) return false;
sr[i]-=fmin[i][j],sc[j]-=fmin[i][j];
sumr-=fmin[i][j],sumc-=fmin[i][j];
fmax[i][j]-=fmin[i][j];
if (fmax[i][j]< || sr[i]< || sc[j]<) return false;
}
return sumr==sumc;
} void addedge(ll U,ll V,ll W)
{
edge++;
to[edge]=V,c[edge]=W,next[edge]=first[U],first[U]=edge;
edge++;
to[edge]=U,c[edge]=,next[edge]=first[V],first[V]=edge;
} bool bfs()
{
Q[bot=top=]=t,tag[t]=++TAG,d[t]=,can[t]=false;
while (bot<=top)
{
ll cur=Q[bot++];
for (ll i=first[cur]; i!=-; i=next[i])
if (c[i^] && tag[to[i]]!=TAG)
{
tag[to[i]]=TAG,d[to[i]]=d[cur]+;
can[to[i]]=false,Q[++top]=to[i];
if (to[i]==s) return true;
}
}
return false;
} ll dfs(ll cur,ll num)
{
if (cur==t) return num;
ll tmp=num,k;
for (ll i=first[cur]; i!=-; i=next[i])
if (c[i] && d[to[i]]==d[cur]- && tag[to[i]]==TAG && !can[to[i]])
{
k=dfs(to[i],min(c[i],num));
if (k) num-=k,c[i]-=k,c[i^]+=k;
if (!num) break;
}
if (num) can[cur]=true;
return tmp-num;
} ll maxflow()
{
ll tot=;
while (bfs()) tot+=dfs(s,~0U>>);
return tot;
} int main()
{
char S[];
ll x,y,z,cas=,up,dn;
scanf("%I64d",&T);
while (T--)
{
scanf("%I64d%I64d",&n,&m);
_init();
for (ll i=; i<=n; i++) scanf("%I64d",&sr[i]),sumr+=sr[i];
for (ll i=; i<=m; i++) scanf("%I64d",&sc[i]),sumc+=sc[i];
scanf("%I64d",&R);
while (R--)
{
scanf("%I64d%I64d%s%I64d",&x,&y,S,&z);
if (S[]=='=') up=z,dn=max(0LL,z);
else if (S[]=='>') up=inf,dn=max(0LL,z+);
else up=z-,dn=;
if (x== && y==)
{
for (ll i=; i<=n; i++)
for (ll j=; j<=m; j++) minsize(i,j,dn,up);
}
else if (x==)
{
for (ll i=; i<=n; i++) minsize(i,y,dn,up);
}
else if (y==)
{
for (ll j=; j<=m; j++) minsize(x,j,dn,up);
}
else minsize(x,y,dn,up);
}
if (cas++) puts("");
if (!check())
{
puts("IMPOSSIBLE");
continue;
}
for (ll i=; i<=n; i++) addedge(s,i,sr[i]);
for (ll j=; j<=m; j++) addedge(n+j,t,sc[j]);
for (ll i=; i<=n; i++)
for (ll j=; j<=m; j++)
if (fmax[i][j]>) addedge(i,n+j,fmax[i][j]);
/*
for (int i=1; i<=n; i++)
{
cout<<" hehe : ";
for (int j=1; j<=m; j++) cout<<fmin[i][j]<<"("<<fmax[i][j]<<") ... ";
cout<<endl;
}
*/
if (maxflow()!=sumr)
{
puts("IMPOSSIBLE");
continue;
}
for (ll i=n+n+m+m; i<=edge; i+=)
{
x=to[i+],y=to[i]-n;
ans[x][y]=c[i+];
}
for (ll i=; i<=n; i++)
{
printf("%I64d",ans[i][]+fmin[i][]);
for (ll j=; j<=m; j++) printf(" %I64d",ans[i][j]+fmin[i][j]);
printf("\n");
}
}
return ;
}
POJ2396_Budget的更多相关文章
随机推荐
- 两个二进制数多少个位(bit)不同
class Solution { public: /** * 获得两个整形二进制表达位数不同的数量 * * @param m 整数m * @param n 整数n * @return 整型 */ in ...
- Python基础篇【第2篇】: Python自定义函数
Python函数 1. 函数定义 函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段.函数能提高应用的模块性,和代码的重复利用率.你已经知道Python提供了许多内建函数,比如print ...
- jquery.easypiechart.js简介
此插件主要是用来统计新的访问.跳出率.服务器负载.使用的RAM等,功能很强大,带有HTML5的动画效果,效果非常炫,看效果吧easyPieChart一款新型的EASY饼图数据统计Jquery插件截图: ...
- 1. Longest Palindromic Substring ( 最长回文子串 )
要求: Given a string S, find the longest palindromic substring in S. (从字符串 S 中最长回文子字符串.) 何为回文字符串? A pa ...
- Tomcat中解决sql server连接失败--- java.lang.ClassNotFoundException: com.microsoft.jdbc.sqlserver.SQLServerDriver
php连接mysql数据库很容易,他俩真是黄金搭档.最近转战java连接微软sqlServer,步骤稍微复杂一点,但也不是太难,中途遇到了一点小问题,最后在csdn论坛里找到了答案http://bbs ...
- Android插件化开发
客户端开发给人的印象往往是小巧,快速奔跑.但随着产品的发展,目前产生了大量的门户型客户端.功能模块持续集成,开发人员迅速增长.不同的开发小组开发不同的功能模块,甚至还有其他客户端集成进入.能做到功能模 ...
- 洛谷P3372 【模板】线段树 1
P3372 [模板]线段树 1 153通过 525提交 题目提供者HansBug 标签 难度普及+/提高 提交 讨论 题解 最新讨论 [模板]线段树1(AAAAAAAAA- [模板]线段树1 洛谷 ...
- csdn的资源使用
资源库: http://lib.csdn.net/
- python类及其方法
python类及其方法 一.介绍 在 Python 中,面向对象编程主要有两个主题,就是类和类实例类与实例:类与实例相互关联着:类是对象的定义,而实例是"真正的实物",它存放了类中 ...
- C# string类型和byte[]类型相互转换
string类型转成byte[]: byte[] byteArray = System.Text.Encoding.Default.GetBytes ( str ); byte[]转成string: ...