算法:网络流(最大流)+并查集(判联通)
思路:
我们发现每一天飞船停靠的地方是不一样的,于是我们很自然佛瑞想到我们可以对每个空间站的每个时间点进行拆点(就相当于是一个动态图,每过一个时间点都可以建一个图)
总结来说就是对于这种一个点(表面意义上的一个点,比如说一个位置)对应多种情况的(比如说随着时间的推移有着不同的状态,而且这种状态>2),我们考虑在类似于分层图上面跑网络流。
接下来是建图:
首先地球是起始站,源点肯定要向每个时刻的地球连边吧,容量INF。然后月球是终点,所以要向每个时刻的月球连边,容量INF。之后对于在空间站的人们就是两种情况:
- 人们可以选择此时在该空间站的飞船,飞向下一个空间站(/地球/月球)。
- 无法移动,所以留在此处。
显然我们把时刻拆开之后,很轻易就能计算出对于每个飞船,当前时刻的位置和下一时刻的位置,直接在分层图上连边即可,容量为最大载重。因为人有可能无法移动,而且空间站的容量为无限大,所以对于每个空间站,要向下一个时刻连一条边,容量INF。
总结来说:
- 源点向每个时刻的地球连容量为INF的边
- 同理,向每个时刻的月球连容量为INF的边
- t时刻的x空间站向t+1时刻的x空间站连一条边(人们可以留在该空间站)
- 若x空间站可以到达y空间站,则从t时刻的x空间站在向t+1时刻的y空间站连一条流量为飞船流量限制的边
最后,当当前的最大流已经超过总人数时,则此时所有人都已经被运到月球,此时输出时间就好了
最后的最后,注意:
- 首先需要判断是否会有答案,即地球到太阳是否连通,在这里用并查集就可以了
- 注意n要+=2,地球和月球也算空间站
这些都理解以后就是一个板子题了!
代码附上:(dinic)
#include<bits/stdc++.h>
using namespace std;
const int maxn=100005;
const int maxm=1005;
const int INF=0x3f3f3f3f;
int n,m,k,Time,Tot,sum,tot=1,S,T=10000;
//S,T即起点,终点
int head[maxn],dep[maxn],cur[maxn],fa[maxn],num[maxn],cap[maxn],to[maxm][maxm];
//没啥好说的
vector<int> v[maxm];
queue<int> q;
struct node
{
int from,to,next;
int val;
}edge[maxn];
inline int read()
{
int s=0,f=1;
char c=getchar();
while (c<'0'||c>'9')
{
if (c=='-')
{
f=-1;
}
c=getchar();
}
while (c>='0'&&c<='9')
{
s=s*10+c-