17.7.11 校内赛 【图论】【最大流】【SPFA】
来源:互联网 发布:sql 查询分析器 编辑:程序博客网 时间:2024/06/08 10:36
1.幻想乡的异变
【题解】
我们先跑一遍SPFA,打出dis数组,为了保证我们能找到其他路径,我们在跑最大流之前先进行的那次dfs中,只要有dis[v]=dis[u]+ed[i].w,就可以认为这个点在某一条最短路上。最后我们再把图上的点建一个反向的,流量为inf的边,跑一遍最大流即可。
代码如下:
#include<cstdio>#include<cstring>#include<algorithm>#include<queue>#define NAME "change"using namespace std;const int N=300;const int M=1e5;const int inf=1e9;int head[N+5],num=1;struct Edge{ int v,next,flow,cap,w;};Edge ed[2*M+5];void build(int i,int j,int w){ ed[++num].v=j; ed[num].next=head[i]; ed[num].w=w; ed[num].flow=0; head[i]=num;}int n,m,a,b,c,dis[N+5],flag[N+5],f[N+5];void spfa(){ queue<int>q; while(!q.empty()) q.pop(); q.push(1); memset(dis,63,sizeof(dis)); dis[1]=0; while(!q.empty()) { int u=q.front();q.pop(); flag[u]=0; for(int i=head[u];i;i=ed[i].next) { int v=ed[i].v; if(dis[v]>dis[u]+ed[i].w) { dis[v]=dis[u]+ed[i].w; if(!flag[v]) { flag[v]=1; q.push(v); } } } }}struct Dinic{ int s,t,d[N+5]; bool bfs() { memset(d,0,sizeof(d)); queue<int>q; q.push(s); d[s]=1; while(!q.empty()) { int u=q.front();q.pop(); for(int i=head[u];i;i=ed[i].next) { int v=ed[i].v; if(ed[i].cap>ed[i].flow&&!d[v]) { d[v]=d[u]+1; q.push(v); } } } if(d[t]) return true; else return false; } int dfs(int u,int a) { if(u==t||a==0)return a; int flow=0,f; for(int i=head[u];i;i=ed[i].next) { int v=ed[i].v; if(d[v]==d[u]+1&&(f=dfs(v,min(a,ed[i].cap-ed[i].flow)))>0) { ed[i].flow+=f; ed[i^1].flow-=f; flow+=f; a-=f; if(a==0) return flow; } } if(!flow)d[u]=-1; return flow; } int maxflow() { int flow=0; while(bfs()) flow+=dfs(s,inf); return flow; }};Dinic date;void dfs(int u){ flag[u]=1; for(int i=head[u];i;i=ed[i].next) { int v=ed[i].v; if(ed[i].w+dis[u]==dis[v]) { ed[i].cap++; ed[i^1].cap=0; if(!flag[v]) { dfs(v); } } }}int main(){ freopen(NAME".in","r",stdin); freopen(NAME".out","w",stdout); scanf("%d%d",&n,&m); date.s=1;date.t=n; for(int i=1;i<=m;i++) { scanf("%d%d%d",&a,&b,&c); build(a,b,c); build(b,a,inf); } spfa(); memset(flag,0,sizeof(flag)); dfs(1); printf("%d\n",date.maxflow()); return 0;}
2.幻想乡的符(hu)卡
【题解】
略,见文帝的博客:
他好像快懂了
3.幻想乡的例大祭
【题解】
首先,存储的时候有一点小技巧,我们存储临界表的时候没必要规规矩矩地按照输入所给的那样,我们把u和v颠倒一下,这样最后就不必讨论2-n到1的路径,只需要1到2-n的的路径就行了。
题目告诉我们,只要是在一个图的双联通分量里的边,路过的时候就要消耗灵力。由此我们先跑一遍Tarjan,求出图的双联通分量。
之后我们再保证灵力最短的情况下,跑一遍SPFA,在灵力最短的路径中找出耗时最少的那一个。
代码如下:
#include<queue>#include<cstdio>#include<cstring> b v #include<algorithm>#define NAME "festival"using namespace std;const int N=1e5;int n,m;int head[N+5],num;long long dis[N+5];long long cost[N+5];int dfn[N+5],low[N+5],con[N+5],sta[N+5],flag[N+5],top,tim; struct edge{ int v,w; int next; edge(){next=-1;}}ed[N+5];void build(int u,int v,int w){ ed[++num].v=v; ed[num].w=w; ed[num].next=head[u]; head[u]=num;}void tarjan(int u){ dfn[u]=low[u]=++tim; flag[sta[++top]=u]=1; for(int i=head[u];i!=-1;i=ed[i].next) { int v=ed[i].v; if(!dfn[v]) { tarjan(v); low[u]=min(low[u],low[v]); } else if(flag[v])low[u]=min(low[u],dfn[v]); } if(dfn[u]==low[u]) for(int now=0;now!= u;) { flag[now=sta[top--]]=0; con[now]=u; }}void SPFA(){ deque<int> q; memset(flag,0,sizeof flag); memset(dis,-1,sizeof dis); memset(cost,-1,sizeof cost); q.push_back(1);dis[1]=0,cost[1]=0; while(!q.empty()) { int u=q.front(); q.pop_front(); flag[u]=0; for(int i=head[u];i!=-1;i=ed[i].next) { int v=ed[i].v; int cc=con[v]==con[u]?cost[u]+1:cost[u]; if(cost[v]>cc||cost[v]==-1||(dis[v]>dis[u]+ed[i].w&&cost[v]==cc)||(dis[v]==-1&&cost[v]==cc)) { cost[v]=cc; dis[v]=dis[u]+ed[i].w; if(!flag[v]) { flag[v]=1; if(!q.empty()) { if(dis[v]>dis[q.front()])q.push_back(v); else q.push_front(v); } q.push_back(v); } } } }}int main(){ freopen(NAME".in","r",stdin); freopen(NAME".out","w",stdout); memset(head,-1,sizeof(head)); scanf("%d%d",&n,&m); for(int i=1;i<=m;i++) { int u,v,w; scanf("%d%d%d",&u,&v,&w); build(v,u,w); } for(int i=1;i<=n;i++) if(!con[i])tarjan(i); SPFA(); for(int k=2;k<=n;k++) { if(dis[k]==dis[0]) printf("-1\n"); else printf("%I64d %I64d\n",cost[k],dis[k]); } return 0;}
以上
2017.7.11
阅读全文
1 0
- 17.7.11 校内赛 【图论】【最大流】【SPFA】
- BZOJ3931【最大流】【SPFA】
- 最小费用最大流,SPFA
- bzoj 3931(spfa+最大流)
- 17.4.5 图论 校内赛
- 校内赛
- 最小费用最大流uva10806+spfa
- Farm Tour-最小费用最大流/SPFA
- hdoj 4240 Route Redundancy 【最大流+ SPFA】
- 最小费用最大流 spfa() + ek()
- poj2195 spfa 最小费用最大流
- 【bzoj3931】【CQOI2015】【网络吞吐量】【spfa+最大流】
- [bzoj3931][CQOI2015]网络吞吐量 spfa+最大流
- 【模板】SPFA增广 最小费用最大流
- hdu1533(最小费用最大流spfa模板)
- 最大流 SAP模板+ 最小费用最大流SPFA模板 +spfa求最短路径
- 2017.7.3 校内赛 【模拟】【贪心】【图论】
- hdu 3416 Marriage Match IV 【图论-网络流-最短路+最大流(spfa + Dinic)】
- 《面向对象程序设计-C++》学习笔记2
- ES的集群名字被谁改了?
- volley学习笔记四
- BZOJ 3262 陌上花开 树状数组套splay
- 2 Add Two Numbers
- 17.7.11 校内赛 【图论】【最大流】【SPFA】
- MySQL优化之expalin语句详解
- 一个细节问题:无法将类 com.tour.info.admin.service.TempService中的方法 indexZtemp应用到给定类型;
- 【二叉树】一棵树的子树是一颗树
- 《软件测试价值提升之路》学习之—测试工程师能力模型
- 道路和航路
- hadoop学习之hadoop完全分布式集群安装
- 小结(数组、循环)
- poj3169Layout【差分约束】 (S)