bzoj1834 网络扩容 网络流
好久没写题解了啊···
题目大意:
给你一幅n个点的网络,先求出其1到n的最大流,每条弧还会有个属性\(cost_i\),表示没扩容一个单位的费用,现在我们要求的就是扩容K个单位的最小费用
思路:
这是一道比较裸的网络流,第一问直接dinic就是了,重点就在于第二问。我们把第一问的残量网络继续利用,其中的每条弧的费用都是0,此时我们再在第\(i\)条弧的两端之间在建一条弧,弧的容量是\(INF\),费用就是\(cost_i\)。这样我们固然可以保证费用正确,可是我们保证不了扩容了K,我们就可以建一个超级源点,连向1号点,容量为K,费用为0。在这个网络中最大流一定是满流,也就是K啊,此时的最小费用就是我们所要求的。
具体实现:
这就很简单了,一遍\(dinic\),再一遍\(MCMF\)我们就可以了
代码如下
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int maxn=10001,maxm=50001,inf=0x7f7f7f7f;
int n,m,k,tot,next[maxm<<1],beg[maxm<<1],head[maxn],flow[maxm<<1],fflow[maxm<<1],last[maxn],pre[maxn],fl[maxn],nxt[maxm<<1],to[maxm<<1],ccost[maxm<<1],cost[maxm<<1],d[maxn],dep[maxn];
bool vis[maxn];
void addedge(int x,int y,int z,int co,int type){
nxt[++tot]=head[x];
head[x]=tot;
to[tot]=y;
beg[tot]=x;
flow[tot]=z;
fflow[tot]=type?z:0;
cost[tot]=type?co:0;
ccost[tot]=co;
}
bool bfs(){
memset(dep,0,sizeof(dep));
queue<int> q;
q.push(1);
dep[1]=1;
while(!q.empty()){
int x=q.front();
q.pop();
for(int i=head[x];i;i=nxt[i]){
int u=flow[i],v=to[i];
if(u>0 and !dep[v]){
dep[v]=dep[x]+1;
q.push(v);
}
}
}
return dep[n];
}
int dfs(int x,int mini){
if(x==n)return mini;
for(int i=head[x];i;i=nxt[i]){
int u=flow[i],v=to[i];
if(u>0 and dep[v]==dep[x]+1){
int dd=dfs(v,min(mini,u));
if(dd>0){
flow[i]-=dd;
flow[i^1]+=dd;
return dd;
}
}
}
return 0;
}
int dinic(){
int ret=0;
while(bfs()){
int tmp=dfs(1,inf);
while(tmp){
ret+=tmp;
tmp=dfs(1,inf);
}
}
return ret;
}
bool spfa() {
memset(d,0x7f,sizeof(d));
memset(fl,0x7f,sizeof(fl));
memset(vis,0,sizeof(vis));
queue<int> q;
q.push(n+1);
vis[n+1]=1;
d[n+1]=0;
pre[n]=-1;
while(!q.empty()) {
int now=q.front();
q.pop();
for(int i=head[now];i;i=nxt[i]){
int v=to[i];
if(fflow[i]>0 and d[v]>d[now]+cost[i]){
d[v]=d[now]+cost[i];
pre[v]=now;
last[v]=i;
fl[v]=min(fl[now],fflow[i]);
if(!vis[v]){
vis[v]=1;
q.push(v);
}
}
}
vis[now]=0;
}
return pre[n]!=-1;
}
int mcmf(){
int ret=0;
while (spfa()){
int now=n;ret+=fl[n]*d[n];
while (now!=n+1){
fflow[last[now]]-=fl[n];
fflow[last[now]^1]+=fl[n];
now=pre[now];
}
}
return ret;
}
void rebuild(){
int cnt=tot;
for(int i=2;i<=cnt;i+=2){
fflow[i]=flow[i];
fflow[i+1]=flow[i+1];
addedge(beg[i],to[i],inf,ccost[i],1);
addedge(to[i],beg[i],0,-ccost[i],1);
}
}
int main(){
scanf("%d%d%d",&n,&m,&k);
addedge(n+1,1,k,0,1);
for(int i=1;i<=m;i++){
int a,b,c,d;
scanf("%d%d%d%d",&a,&b,&c,&d);
addedge(a,b,c,d,0);
addedge(b,a,0,-d,0);
}
int ans1=dinic();
rebuild();
int ans2=mcmf();
printf("%d %d\n",ans1,ans2);
return 0;
}
bzoj1834 网络扩容 网络流的更多相关文章
- BZOJ-1834 网络扩容 最小费用最大流+最大流+乱搞
1834: [ZJOI2010]network 网络扩容 Time Limit: 3 Sec Memory Limit: 64 MB Submit: 2269 Solved: 1136 [Submit ...
- [ZJOI2010][bzoj1834] 网络扩容 [费用流]
题面 传送门 思路 第一问:无脑网络流跑一波 第二问: 先考虑一个贪心的结论:扩容出来的扩容流量一定要跑满 证明显然 因此我们可以把扩容费用可以换个角度思考,变成增加一点流量,花费W的费用 这样,我们 ...
- bzoj1834 网络扩容
Description 给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用.求: 1. 在不扩容的情况下,1到N的最大流: 2. 将1到N的最大流增加K所需的 ...
- 洛谷$P2604\ [ZJOI2010]$网络扩容 网络流
正解:网络流 解题报告: 传送门$QwQ$ 昂第一问跑个最大流就成不说$QwQ$ 然后第二问,首先原来剩下的边就成了费用为0的边?然后原来的所有边连接的两点都给加上流量为$inf$费用为$w$的边,保 ...
- 【BZOJ1834】网络扩容(最大流,费用流)
[BZOJ1834]网络扩容(最大流,费用流) 题面 Description 给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用.求: 1. 在不扩容的情况下 ...
- 【BZOJ1834】[ZJOI2010]network 网络扩容 最大流+最小费用流
[BZOJ1834][ZJOI2010]network 网络扩容 Description 给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用.求: 1. 在不 ...
- 【bzoj1834】[ZJOI2010]network 网络扩容
1834: [ZJOI2010]network 网络扩容 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 2701 Solved: 1368[Submit ...
- [BZOJ1834][ZJOI2010]network 网络扩容 最大流+费用流
1834: [ZJOI2010]network 网络扩容 Time Limit: 3 Sec Memory Limit: 64 MB Submit: 3330 Solved: 1739 [Subm ...
- bzoj1834: [ZJOI2010]network 网络扩容
努力看了很久样例一直过不了...然后各种输出中间过程啊巴拉巴拉弄了1h,没办法了...然后突然想到啊原来的边可以用啊为什么不用...于是A了...感人肺腑 #include<cstdio> ...
随机推荐
- OC中protocol、category和继承的关系--转
开放封闭原则(OCP)就是,“对扩展开放,对更改封闭”.是所有面向对象设计的一个核心宗旨.感兴趣的可以看百度百科的一些解释:http://baike.baidu.com/view/2493421.ht ...
- windows下Python的安装,以及IDLE的使用
一.Python的下载安装 (1)python的windows安装包可以从https://www.python.org 网址中下载,可以下载3.4版本的或者2.7版本的.(2)下载后直接运行即可.然后 ...
- E. The Values You Can Make 背包,同时DP
http://codeforces.com/problemset/problem/688/E 题目需要在n个数中找出一个集合,使得这个集合的和为val,然后问这些所有集合,能产生多少个不同的和值. 题 ...
- [转]Asp.net MVC中Html.Partial, RenderPartial, Action,RenderAction 区别和用法
本文转自:http://www.cnblogs.com/gesenkof99/archive/2013/06/03/3115052.html Partial 和RenderPartial:这两个的性质 ...
- 7.JAVA-类继承、覆写、final关键字
1.JAVA继承-extends 在java中,要想实现继承则使用extends关键字. 一般子类被称为派生类,父类称为基类(super) extends需要注意的地方: java不允许多重继承(一个 ...
- scala.的Enumeration枚举示例(转)
简介 在scala中没有枚举类型,但在标准类库中提供了Enumeration类来产出枚举.扩展Enumeration类后,调用value方法类初始化枚举中的可能值. 内部类value实际上是一个抽象类 ...
- SQL Server中行列转置方法
PIVOT用于将列值旋转为列名(即行转列),在SQL Server 2000可以用聚合函数配合CASE语句实现 PIVOT的一般语法是:PIVOT(聚合函数(列) FOR 列 in (…) )AS P ...
- pythno学习小结-替换python字典中的key值
源: d={'a':1,'b':2,'c':3} 目标:key:'b'替换为'e' d={'a':1,'e':2,'c':3} 方法: d['e']=d.pop('b')
- Java C
先说一下自己叫什么,免得面试的人张冠李戴. 介绍自己有几个方面:1学什么专业的那方面学的过硬,可以说的具体点. 2以前做过什么.(这家公司要你肯定是和你的经历有关.) 3现在来这家公司的目的是什么(当 ...
- 多线程之Java中的等待唤醒机制
多线程的问题中的经典问题是生产者和消费者的问题,就是如何让线程有序的进行执行,获取CPU执行时间片的过程是随机的,如何能够让线程有序的进行,Java中提供了等待唤醒机制很好的解决了这个问题! 生产者消 ...