zoj 3229 上下界网络最大可行流带输出方案
收获:
1. 上下界网络流求最大流步骤:
1) 建出无环无汇的网络,并看是否存在可行流
2) 如果存在,那么以原来的源汇跑一次最大流
3) 流量下界加上当前网络每条边的流量就是最大可行流了.
2. 输出方案:
可以把边的位置信息一起存在边表中,求完最大流后遍历一下边,把信息更新过去.
#include <cstdio>
#include <cstring>
#define min(a,b) ((a)<(b)?(a):(b))
#define oo 0x3f3f3f3f
#define N 1500
#define M 500000 struct Dinic {
int n, src, dst;
int head[N], dest[M], flow[M], next[M], info[M][], etot;
int cur[N], dep[N], qu[N], bg, ed;
void init( int n, int src, int dst ) {
this->n = n;
this->src = src;
this->dst = dst;
etot = ;
memset( head, -, sizeof(head) );
}
void adde( int i, int j, int u, int v, int f ) {
next[etot]=head[u],flow[etot]=f,dest[etot]=v,info[etot][]=i,info[etot][]=j; head[u]=etot++;
next[etot]=head[v],flow[etot]=,dest[etot]=u,info[etot][]=-,info[etot][]=-; head[v]=etot++;
}
bool bfs() {
memset( dep, , sizeof(dep) );
qu[bg=ed=] = src;
dep[src] = ;
while( bg<=ed ) {
int u=qu[bg++];
for( int t=head[u]; t!=-; t=next[t] ) {
int v=dest[t], f=flow[t];
if( f && !dep[v] ) {
dep[v] = dep[u]+;
qu[++ed] = v;
}
}
}
return dep[dst];
}
int dfs( int u, int a ) {
if( u==dst || a== ) return a;
int remain=a, past=, na;
for( int &t=cur[u]; t!=-; t=next[t] ) {
int v=dest[t], &f=flow[t], &vf=flow[t^];
if( f && dep[v]==dep[u]+ && (na=dfs(v,min(remain,f))) ) {
f -= na;
vf += na;
remain -= na;
past += na;
if( !remain ) break;
}
}
return past;
}
int maxflow() {
int f=;
while( bfs() ) {
memcpy( cur, head, sizeof(cur) );
f += dfs(src,oo);
}
return f;
}
}D;
struct Btop {
int n;
int head[N], dest[M], bval[M], tval[M], next[M], info[M][], etot;
int si[N], so[N];
void init( int n ) {
this->n = n;
etot = ;
memset( head, -, sizeof(head) );
memset( si, , sizeof(si) );
memset( so, , sizeof(so) );
}
void adde( int i, int j, int u, int v, int b, int t ) {
next[etot]=head[u],tval[etot]=t,bval[etot]=b,dest[etot]=v;
info[etot][]=i, info[etot][]=j;
si[v] += b, so[u] += b;
head[u]=etot++;
}
bool ok() {
int src=n+, dst=n+;
D.init( dst, src, dst );
for( int u=; u<=n; u++ )
for( int t=head[u]; t!=-; t=next[t] ) {
int v=dest[t];
D.adde( info[t][], info[t][], u, v, tval[t]-bval[t] );
}
int sum = ;
for( int u=; u<=n; u++ ) {
if( si[u]>so[u] ) {
D.adde( -, -, src, u, si[u]-so[u] );
sum += si[u]-so[u];
} else if( so[u]>si[u] ) {
D.adde( -, -, u, dst, so[u]-si[u] );
}
}
return sum==D.maxflow();
}
}B; int n, m;
int c[];
int ans[][], tot; int main() {
while( scanf( "%d%d", &n, &m ) == ) {
int src=n+m+, dst=src+;
B.init(dst);
for( int i=,g; i<=m; i++ ) {
scanf( "%d", &g );
B.adde( -, -, n+i, dst, g, oo );
}
memset( ans, , sizeof(ans) );
tot = ;
for( int i=,d; i<=n; i++ ) {
scanf( "%d%d", c+i, &d );
B.adde( -, -, src, i, , d );
for( int j=,t,l,r; j<=c[i]; j++ ) {
scanf( "%d%d%d", &t, &l, &r );
t++;
B.adde( i, j, i, n+t, l, r );
ans[i][j] = l;
tot += l;
}
}
B.adde( -, -, dst, src, , oo );
bool ok = B.ok();
if( !ok ) {
printf( "-1\n" );
} else {
D.src=src;
D.dst=dst;
D.maxflow();
for( int t=; t<D.etot; t++ ) {
int i=D.info[t][], j=D.info[t][];
if( ~i && ~j ) {
ans[i][j] += D.flow[t^];
tot += D.flow[t^];
}
}
printf( "%d\n", tot );
for( int i=; i<=n; i++ )
for( int j=; j<=c[i]; j++ )
printf( "%d\n", ans[i][j] );
}
printf( "\n" );
}
}
zoj 3229 上下界网络最大可行流带输出方案的更多相关文章
- sgu 176 上下界网络流最小可行流带输出方案
		算法步骤: 1. 先将原图像最大可行流那样变换,唯一不同的是不加dst->src那条边来将它变成无源无汇的网络流图.直接跑一边超级源到超级汇的最大流. 2. 加上刚才没有加上的那条边p 3. 再 ... 
- POJ2396 Budget(有源汇流量有上下界网络的可行流)
		题目大概给一个有n×m个单元的矩阵,各单元是一个非负整数,已知其每行每列所有单元的和,还有几个约束条件描述一些单元是大于小于还是等于某个数,问矩阵可以是怎样的. 经典的流量有上下界网络流问题. 把行. ... 
- ZOJ2314 Reactor Cooling(无源汇流量有上下界网络的可行流)
		题目大概说一个核反应堆的冷却系统有n个结点,有m条单向的管子连接它们,管子内流量有上下界的要求,问能否使液体在整个系统中循环流动. 本质上就是求一个无源汇流量有上下界的容量网络的可行流,因为无源汇的容 ... 
- ZOJ 1314 Reactor Cooling | 上下界无源汇可行流
		ZOJ 1314 Reactor Cooling | 上下界无源汇可行流 题意 有一个网络,每条边有流量的上界和下界,求一种方案,让里面的流可以循环往复地流动起来. 题解 上下界无源汇可行流的模型: ... 
- HDU3157 Crazy Circuits(有源汇流量有上下界网络的最小流)
		题目大概给一个电路,电路上有n+2个结点,其中有两个分别是电源和负载,结点们由m个单向的部件相连,每个部件都有最少需要的电流,求使整个电路运转需要的最少电流. 容量网络的构建很容易,建好后就是一个有源 ... 
- hdu 4940 Destroy Transportation system( 无源汇上下界网络流的可行流推断 )
		题意:有n个点和m条有向边构成的网络.每条边有两个花费: d:毁坏这条边的花费 b:重建一条双向边的花费 寻找这样两个点集,使得点集s到点集t满足 毁坏全部S到T的路径的费用和 > 毁坏全部T到 ... 
- BZOJ 3876 支线剧情  有源汇有上下界最小费用可行流
		题意: 给定一张拓扑图,每条边有边权,每次只能从第一个点出发沿着拓扑图走一条路径,求遍历所有边所需要的最小边权和 分析: 这道题乍一看,可能会想到什么最小链覆盖之类的,但是仔细一想,会发现不行,一是因 ... 
- BZOJ 2055 80人环游世界 有上下界最小费用可行流
		题意: 现在有这么一个m人的团伙,也想来一次环游世界. 他们打算兵分多路,游遍每一个国家. 因为他们主要分布在东方,所以他们只朝西方进军.设从东方到西方的每一个国家的编号依次为1...N.假若第 ... 
- ZOJ3229 Shoot the Bullet(有源汇流量有上下界网络的最大流)
		题目大概说在n天里给m个女孩拍照,每个女孩至少要拍Gi张照片,每一天最多拍Dk张相片且都有Ck个拍照目标,每一个目标拍照的张数要在[Lki, Rki]范围内,问最多能拍几张照片. 源点-天-女孩-汇点 ... 
随机推荐
- 【codeforces】【比赛题解】#960 CF Round #474 (Div. 1 + Div. 2, combined)
			终于打了一场CF,不知道为什么我会去打00:05的CF比赛…… 不管怎么样,这次打的很好!拿到了Div. 2选手中的第一名,成功上紫! 以后还要再接再厉! [A]Check the string 题意 ... 
- mysql 在windons下的备份命令
			1. @echo off set "Ymd=%date:~,4%%date:~5,2%%date:~8,2%" mysqldump -uroot -proot jy510 > ... 
- Shell-help格式详解
			前言 linux shell命令通常可以通过-h或--help来打印帮助说明,或者通过man命令来查看帮助,有时候我们也会给自己的程序写简单的帮助说明,其实帮助说明格式是有规律可循的 帮助示例 下面是 ... 
- Method for balancing binary search trees
			Method for balancing a binary search tree. A computer implemented method for balancing a binary sear ... 
- 打包egg
			scrapyd-deploy -p chahao -v 1.0 --build-egg chahao.egg 
- 用jquery的ajax方法获取不到return返回值
			如果jquery中,获取不到ajax返回值. 两个错误写法会导致这种情况:1.ajax未用同步 2.在ajax方法中直接return返回值. 下面列举了三种写法,如果想成功获取到返回值,参考第三种写法 ... 
- 转:vue-cli的webpack模板项目配置文件分析
			转载地址:http://blog.csdn.net/hongchh/article/details/55113751 一.文件结构 本文主要分析开发(dev)和构建(build)两个过程涉及到的文件, ... 
- 正则表达式之你不知道的replace
			我们都知道js内置类String提供了一个replace方法,可以把指定字符串替换为另一个字符串,他的基础用法如下: var str="hello world"; var str= ... 
- 简单ORACLE分区表、分区索引
			前一段听说CSDN.COM里面很多好东西,同事建议看看合适自己也可以写一写,呵呵,今天第一次开通博客,随便写点东西,就以第一印象分区表简单写第一个吧. ORACLE对于分区表方式其实就是将表分段存储, ... 
- OR 连接查询注意
			用or 查询时, 取得是 每个or中条件的 查询的结果集union. select * from categorysecond t where ISNULL(null); ort.csid in (' ... 
