传送门

题意

给你一个 $ n $ 个点,$ m $ 条边的无向网络,每条边有长度。每个点的流量限制为 $ c[i] $ 。

要求流量只能经过从 $ 1 $ 的 $ n $ 的最短路。问你最大流是多少。

题解

先以 $ 1 $ 和 $ n $ 分别为起点跑一遍dijkstra,判断出哪些边是在最短路上的。

将每个点 $ i $ 拆成两个点 $ A(i), B(i) $ ,从 $ A(i) $ 向 $ B(i) $ 连一条容量为 $ c[i] $ 的边。

对于每条在最短路上的边 $ i \to j $ ,从 $ B(i) $ 向 $ A(j) $ 连一条容量为 $ INF $ 的边。

最后从 $ B(1) $ 到 $ A(n) $ 跑一遍最大流就好。

AC Code

#include <iostream>
#include <stdio.h>
#include <string.h>
#include <vector>
#include <queue>
#define MAX_N 1005
#define MAX_M 100005
#define INF 100000000000000ll
#define A(x) (x)
#define B(x) (n+(x))
#define int long long using namespace std; struct Ed
{
int dst,len;
Ed(int _dst,int _len) { dst=_dst,len=_len; }
Ed(){}
}; struct Data
{
int x,w;
Data(int _x,int _w) { x=_x,w=_w; }
Data(){}
friend bool operator < (const Data &a,const Data &b)
{
return a.w>b.w;
}
}; struct Edge
{
int dst,cap,rev;
Edge(int _dst,int _cap,int _rev) { dst=_dst,cap=_cap,rev=_rev; }
Edge(){}
}; int n,m,tot;
int a[MAX_M];
int b[MAX_M];
int c[MAX_M];
int d[MAX_N];
int dis[MAX_N];
int rev[MAX_N];
bool vis[MAX_N];
int it[MAX_N];
int lv[MAX_N];
vector<Ed> ed[MAX_N];
priority_queue<Data> Q;
vector<Edge> edge[MAX_N];
queue<int> q; void read()
{
scanf("%lld%lld",&n,&m);
for(int i=1;i<=m;i++)
{
scanf("%lld%lld%lld",&a[i],&b[i],&c[i]);
ed[a[i]].push_back(Ed(b[i],c[i]));
ed[b[i]].push_back(Ed(a[i],c[i]));
}
for(int i=1;i<=n;i++) scanf("%lld",&d[i]);
} void dij(int s,int *dis)
{
memset(dis+1,0x3f,sizeof(int)*n);
memset(vis+1,false,sizeof(bool)*n);
dis[s]=0,Q.push(Data(s,0));
while(!Q.empty())
{
int x=Q.top().x; Q.pop();
if(vis[x]) continue; vis[x]=true;
for(int i=0;i<ed[x].size();i++)
{
Ed temp=ed[x][i];
if(dis[temp.dst]>dis[x]+temp.len)
{
dis[temp.dst]=dis[x]+temp.len;
Q.push(Data(temp.dst,dis[temp.dst]));
}
}
}
} inline void add(int s,int t,int c)
{
edge[s].push_back(Edge(t,c,edge[t].size()));
edge[t].push_back(Edge(s,0,edge[s].size()-1));
} void build()
{
tot=(n<<1);
for(int i=2;i<n;i++) add(A(i),B(i),d[i]);
for(int i=1;i<=m;i++)
{
if(dis[a[i]]+c[i]+rev[b[i]]==dis[n]) add(B(a[i]),A(b[i]),INF);
if(dis[b[i]]+c[i]+rev[a[i]]==dis[n]) add(B(b[i]),A(a[i]),INF);
}
} void bfs(int s)
{
memset(lv+1,0,sizeof(int)*tot);
lv[s]=1,q.push(s);
while(!q.empty())
{
int x=q.front(); q.pop();
for(int i=0;i<edge[x].size();i++)
{
Edge temp=edge[x][i];
if(temp.cap>0 && !lv[temp.dst])
{
lv[temp.dst]=lv[x]+1;
q.push(temp.dst);
}
}
}
} int dfs(int x,int t,int f)
{
if(x==t) return f;
for(int &i=it[x];i<edge[x].size();i++)
{
Edge &temp=edge[x][i];
if(temp.cap>0 && lv[x]<lv[temp.dst])
{
int d=dfs(temp.dst,t,min(f,temp.cap));
if(d>0)
{
temp.cap-=d;
edge[temp.dst][temp.rev].cap+=d;
return d;
}
}
}
return 0;
} int max_flow(int s,int t)
{
int ans=0,f;
while(true)
{
bfs(s);
if(!lv[t]) return ans;
memset(it+1,0,sizeof(int)*tot);
while((f=dfs(s,t,INF))>0) ans+=f;
}
} void work()
{
dij(1,dis);
dij(n,rev);
build();
printf("%lld\n",max_flow(B(1),A(n)));
} signed main()
{
read();
work();
}

BZOJ 3931 [CQOI2015]网络吞吐量:最大流【拆点】的更多相关文章

  1. BZOJ 3931: [CQOI2015]网络吞吐量 最大流

    3931: [CQOI2015]网络吞吐量 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/p ...

  2. BZOJ 3931: [CQOI2015]网络吞吐量( 最短路 + 最大流 )

    最短路 + 最大流 , 没什么好说的... 因为long long WA 了两次.... ------------------------------------------------------- ...

  3. BZOJ 3931: [CQOI2015]网络吞吐量

    3931: [CQOI2015]网络吞吐量 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1555  Solved: 637[Submit][Stat ...

  4. bzoj 3931: [CQOI2015]网络吞吐量 -- 最短路+网络流

    3931: [CQOI2015]网络吞吐量 Time Limit: 10 Sec  Memory Limit: 512 MB Description 路由是指通过计算机网络把信息从源地址传输到目的地址 ...

  5. bzoj 3931 [CQOI2015]网络吞吐量(最短路,最大流)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3931 [题意] 只能通过1-n的最短路,求网络最大流 [思路] 分别以1,n为起点做最 ...

  6. BZOJ 3931: [CQOI2015]网络吞吐量 Dijkstra+最大流

    这个没啥难的. 只保留可以转移最短路的边,然后拆点跑一个最大流即可. #include <bits/stdc++.h> #define N 1004 #define M 250004 #d ...

  7. ●BZOJ 3931 [CQOI2015]网络吞吐量

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=3931 题解: 在最短路图上跑网络流,要开long long(无奈 BZOJ AC 不了,洛谷 ...

  8. 3931: [CQOI2015]网络吞吐量

    3931: [CQOI2015]网络吞吐量 链接 分析: 跑一遍dijkstra,加入可以存在于最短路中的点,拆点最大流. 代码: #include<cstdio> #include< ...

  9. 【BZOJ3931】[CQOI2015]网络吞吐量 最大流

    [BZOJ3931][CQOI2015]网络吞吐量 Description 路由是指通过计算机网络把信息从源地址传输到目的地址的活动,也是计算机网络设计中的重点和难点.网络中实现路由转发的硬件设备称为 ...

随机推荐

  1. 名义人均GDP的背后,中国真实的人均GDP是1.2万美元!(中国GDP含金量较高)

    来源:天涯社区 根据IMF(国际货币基金组织)在今年4月的报告,2014年份中国人均GDP为7600美元,在185个国家当中排行第78位. 然而,根据楼主在国外行走多年的经验,巴西.墨西哥.马来西亚. ...

  2. [NOIP2018TG]保卫王国

    [NOIP2018TG]保卫王国 BZOJ luogu 当动态dp模板题写的,(全集-最大点权独立集)不能放军队的+inf,必须放军队-inf即可 注意矩阵乘法的顺序问题 #define ll lon ...

  3. 3.Write Scripts for the mongo Shell-官方文档摘录

    总结 1 使用js进行获取数据的方法 2 js方式和原生mongo shell的交互方式的区别写法 3 需要将所有数据打印出来使用到的循环示例 cursor = db.collection.find( ...

  4. CodeForces 215B Olympic Medal(数学啊)

    题目链接:http://codeforces.com/problemset/problem/215/B Description The World Programming Olympics Medal ...

  5. 六顶思维帽的思考,敏捷开发?——By Me

    人类的思维可以分为很多种,其中按照思维的深度和广度的侧重,可以分为纵向思维和横向思维两种: 简单的来说,“六顶思维帽”可以简单的理解为下图所示: 如何使用这种思维方式呢?举个例子:先输入一个待讨论的事 ...

  6. 01 javaSe 01 抽象类和接口

      抽象类 接口   目录(?)[-] 1 抽象类与接口是面向对象思想层面概念不是程序设计语言层面概念 2 抽象类是本体的抽象接口是行为的抽象 3 C中抽象类与接口的探讨     目录(?)[+]   ...

  7. [设计模式]State模式

    <Java与模式> 又称状态对象模式.状态模式是对象的行为模式.GOF95 一个对象的行为取决于一个或者多个动态变化的属性,这样的属性叫做状态.这样的对象叫做有状态的对象(stateful ...

  8. 编程语言的介绍(Day2)

    1.什么是编程,为什么要编程? 编程==编写程序(写代码) 2.编程语言有哪些 机器语言 优点是最底层,速度最快,缺点是最复杂,开发效率最低 汇编语言 优点是比较底层,速度最快,缺点是复杂,开发效率最 ...

  9. Visual Studio 2012的Windows Service服务安装方式

    windows服务应用程序是一种长期运行在操作系统后台的程序,它对于服务器环境特别适合,它没有用户界面,不会产生任何可视输出,任何用户输出都回被写进windows事件日志.计算机启动时,服务会自动开始 ...

  10. Java面向对象—多态

    概述:同一个事物,在不同的时刻表现出不同的状态. 代码中如何体现: 要有继承, 要有方法重写, 父类引用指向子类对象 多态的成员访问特点 成员变量:编译看左边(父类), 运行看左边 成员方法:编译看左 ...