BZOJ 1497 JZYZOJ 1344 [NOI2006]最大获利 网络流 最大权闭合图

http://www.lydsy.com/JudgeOnline/problem.php?id=1497

http://172.20.6.3/Problem_Show.asp?id=1344

思路:(最大权闭合图的思路相同)

将所有的用户群获利(正值)作为一个点连一条权值为获利值的边到st点,将所有的建站消耗(输入的是正值但是是在获利中减去的所以实质还是负值)作为一个点连一条权值为消耗值的边到ed点,再将每个用户群点和其依赖的建站点连一条权值为无穷的边,求st到ed的最大流。

此时,所求的最大获利=所有用户群获利的和-最大流。

某条st到ed的路如果得不偿失,贡献的值就是用户群获利的值;否则,贡献值为建站消耗,从而起到了选择的作用。

算是网络流的复习,用奇怪的优化过的dinic才不会超时,其他的方法算最大流都是80分。

dinic的写法和我以前看到的不一样,减少了return次数从而减少了dfs的次数,让每次dfs的值就是目前图中所有路(不重合)能得到的值,大大节省了时间。

代码

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<iostream>
 4 #include<algorithm>
 5 #include<cmath>
 6 #include<queue>
 7 #include<vector>
 8 using namespace std;
 9 const int maxn=55010;
10 const int minf=1<<30;
11 int n,m;
12 int a[maxn]={};
13 int vis[maxn]={};
14 struct nod{
15     int y,next,v,rev;
16 }e[maxn*10];
17 int head[maxn],dep[maxn],tot=0;
18 void init(int x,int y,int v){
19     e[++tot].y=y;e[tot].v=v;e[tot].next=head[x],e[tot].rev=tot+1;
20     head[x]=tot;
21     e[++tot].y=x;e[tot].v=0;e[tot].next=head[y],e[tot].rev=tot-1;
22     head[y]=tot;
23 }
24 int bfs(int st,int ed){
25     queue<int>q;
26     memset(dep,-1,sizeof(dep));
27     dep[st]=0;q.push(st);
28     int x,y,v;
29     while(!q.empty()){
30         x=q.front();q.pop();
31         for(int i=head[x];i;i=e[i].next){
32             y=e[i].y;v=e[i].v;
33             if(dep[y]==-1&&v){
34                 dep[y]=dep[x]+1;q.push(y);
35             }
36         }
37     }
38     return dep[ed]!=-1;
39 }
40 int dfs(int x,int ed,int mi){
41     if(x==ed)return mi;
42     int y,v,f,tsn=0;
43     for(int i=head[x];i;i=e[i].next){
44         y=e[i].y;v=e[i].v;
45         if(v&&dep[y]==dep[x]+1){
46             f=dfs(y,ed,min(mi-tsn,v));
47             e[i].v-=f;
48             e[e[i].rev].v+=f;
49             tsn+=f;
50             if(tsn==mi)return tsn;
51         }
52     }
53     if(!tsn)dep[x]=-1;
54     return tsn;
55
56 }
57 int dinic(int st,int ed){
58     int ans=0;
59     while(bfs(st,ed)){
60         ans+=dfs(st,ed,minf);
61     }
62     return ans;
63 }
64 int main(){
65     scanf("%d%d",&n,&m);
66     int x,y,v,st=n+m+1,ed;
67     ed=st+1;
68     int ans=0;
69     for(int i=1;i<=n;i++){
70         scanf("%d",&a[i]);
71         init(i,ed,a[i]);
72     }
73     for(int i=1;i<=m;i++){
74         scanf("%d%d%d",&x,&y,&v);
75         ans+=v;
76         init(i+n,x,minf);
77         init(i+n,y,minf);
78         init(st,i+n,v);
79     }
80     printf("%d\n",ans-dinic(st,ed));
81     return 0;
82 }

时间: 2024-10-09 20:30:56

BZOJ 1497 JZYZOJ 1344 [NOI2006]最大获利 网络流 最大权闭合图的相关文章

bzoj 1497 [NOI2006]最大获利【最大权闭合子图+最小割】

不要被5s时限和50000点数吓倒!大胆网络流!我一个5w级别的dinic只跑了1s+! 看起来没有最大权闭合子图的特征--限制,实际上还是有的. 我们需要把中转站看成负权点,把p看成点权,把客户看成正权点,把c看成点权,然后把中转站点a.b作为客户点的依赖点 s点向所有正权点连边,流量为点权:所有负权点向t连边,流量为负点权(即正数!) 对于所有有依赖关系的点,由客户点向中转站点连边,流量为inf,也就是最大权闭合子图中的向其依赖点连边 连边的意义详见:http://www.cnblogs.c

1497: [NOI2006]最大获利(最大权闭合子图)

1497: [NOI2006]最大获利 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 5503  Solved: 2673 Description 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU集团旗下的CS&T通讯公司在新一代通讯技术血战的前夜,需要做太多的准备工作,仅就站址选择一项,就需要完成前期市场研究.站址勘测.最优化等项目.在前期市场调查和站址勘测之后,公司得到了一共N个可以作为通讯信号中转站的地址,而由于这

1497. [NOI2006]最大获利【最大权闭合子图】

Description 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU集团旗下的CS&T通讯公司在新一代通讯技术血战的前夜,需要做太多的准备工作,仅就站址选择一项,就需要完成前期市场研究.站址勘测.最优化等项目.在前期市场调查和站址勘测之后,公司得到了一共N个可以作为通讯信号中转站的地址,而由于这些地址的地理位置差异,在不同的地方建造通讯中转站需要投入的成本也是不一样的,所幸在前期调查之后这些都是已知数据:建立第i个通讯中转站需要的成本为Pi(1≤i≤N).另外公

NOI2006 最大获利(最大权闭合子图)

codevs 1789 最大获利 2006年NOI全国竞赛 时间限制: 2 s 空间限制: 128000 KB 题目描述 Description 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是 挑战.THU 集团旗下的 CS&T 通讯公司在新一代通讯技术血战的前夜,需要做 太多的准备工作,仅就站址选择一项,就需要完成前期市场研究.站址勘测.最 优化等项目. 在前期市场调查和站址勘测之后,公司得到了一共 N 个可以作为通讯信号中 转站的地址,而由于这些地址的地理位置差异,在不同

POJ 2987 Firing 网络流 最大权闭合图

http://poj.org/problem?id=2987 https://blog.csdn.net/u014686462/article/details/48533253 给一个闭合图,要求输出其最大权闭合图的权值和需要选的最少点数,最大权闭合图定义和网络流连边方式见博客. 1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 #include<cmath> 5 #include&

【暖*墟】#网络流# 最大权闭合子图

[相关概念详解] 闭合图:有向图的一个点集,且这个点集的所有出边仍然指向该点集. 最大权闭合图:(每一个点有一个权值)在所有的合法闭合图中,点权之和最大的图. 处理问题:权值有正有负,重复选只算一次,选择有相互关联性 的问题. 首先有一个有向连通图(闭合图),每个点带有一个权值,例如: 造出一个超级源点S和一个超级汇点T,把S连边到所有带有正权的点上,每条边的容量是这个点的权: 把所有带负权的点连边到T,每条边的容量是这个点的权的相反数(正值).原来的边的容量设成无限大. 所有的点按权值的正负连

最大权闭合图 &amp;&amp; 【BZOJ】1497: [NOI2006]最大获利

最大权闭合图详细请看胡伯涛论文<最小割模型在信息学竞赛中的应用>,我在这里截图它的定义以及一些东西. 假设我们有一个图,点集的出边都是连到点集的,那么称这个为闭合图.现在这些点集都有个权值,我们要选择某个闭合图使得权值最大. 回到此题: 最大获利这一题,我们可以这样看,用户群和中转站为带权的点集,用户群的权为收益,中转站的权为负的成本,即0-成本,用户群向其中两个中转站连弧,那么这个就是一个闭合图. 我们要求这个闭合图的权值和最大,即最大收益,那么就能转移到上面的求最大权闭合图的做法去了. 做

BZOJ 4873 寿司餐厅(最大权闭合图 网络流)

寿司餐厅 时间限制: 1 Sec  内存限制: 512 MB提交: 6  解决: 3[提交][状态][讨论版] 题目描述 Kiana 最近喜欢到一家非常美味的寿司餐厅用餐.每天晚上,这家餐厅都会按顺序提供n种寿司,第i种寿司有一个代号 ai和美味度 di,i,不同种类的寿司有可能使用相同的代号.每种寿司的份数都是无限的,Kiana 也可以无限次取寿司来吃,但每种寿司每次只能取一份,且每次取走的寿司必须是按餐厅提供寿司的顺序连续的一段,即 Kiana 可以一次取走第 1,2种寿司各一份,也可以一次

BZOJ 1565 植物大战僵尸(最大权闭合图)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1565 题意:植物大战僵尸,一个n*m的格子,每 个格子里有一个植物,每个植物有两个属性:(1)价值:(2)保护集合,也就是这个植物可以保护矩阵中的某些格子.现在你是僵尸,你每次只能从(i,m) 格子进入,从右向左进攻.若一个格子是被保护的那么你是不能进入的.每进入一个格子则吃掉该格子的植物并得到其价值(价值有可能是负的).注意,每次在进 入一行后还可以再退到最右侧然后再换一行吃别的.问