CTSC 1999 家园 【网络流24题】星际转移
直接把每一个点,每一天拆成一个点。
然后每个点到下一天连$inf$的边。
然后把飞船的路径用容量为飞船容量的边连接。
然后跑网络流判断是否满流。
#include <queue>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define inf 0x3f3f3f3f
#define maxn 500005
int n,m,k,p[25],flag=0,S=0,T=maxn-1,ans=0;
int sta[25][25],num[25];
int h[maxn],to[maxn],ne[maxn],fl[maxn],fr[maxn],en=0;
int hash[25][500],cnt=0,dis[maxn];
queue <int> q;
void add(int a,int b,int c)
{
to[en]=b;fr[en]=a;ne[en]=h[a];fl[en]=c;h[a]=en++;
to[en]=a;fr[en]=b;ne[en]=h[b];fl[en]=0;h[b]=en++;
}
bool tell()
{
memset(dis,-1,sizeof dis);
while (!q.empty()) q.pop();
dis[S]=0;q.push(S);
while (!q.empty())
{
int x=q.front();q.pop();
for (int i=h[x];i>=0;i=ne[i])
{
if (fl[i]>0&&dis[to[i]]==-1)
{
dis[to[i]]=dis[x]+1;
q.push(to[i]);
}
}
}
if (dis[T]==-1) return false;
return true;
}
int zeng(int k,int now)
{
if (k==T) return now;
int r=0;
for (int i=h[k];i>=0&&now>r;i=ne[i])
if (dis[k]+1==dis[to[i]]&&fl[i]>0)
{
int t=zeng(to[i],min(now-r,fl[i]));
fl[i]-=t;fl[i^1]+=t;r+=t;
}
if (!r) dis[k]=-1;
return r;
}
int main()
{
memset(h,-1,sizeof h);
scanf("%d%d%d",&n,&m,&k);
F(i,1,m)
{
scanf("%d",&p[i]);
scanf("%d",&num[i]);
F(j,0,num[i]-1)
{
scanf("%d",&sta[i][j]);
if (sta[i][j]==-1) sta[i][j]=n+1;
}
}
F(i,0,n+1)
F(j,0,205)
hash[i][j]=++cnt;
add(S,hash[0][0],k);
add(hash[n+1][0],T,inf);
for (int z=0;z<=200;++z)
{
int tmp;
F(i,0,n+1)
add(hash[i][z],hash[i][z+1],inf);
F(i,1,m)
add(hash[sta[i][z%num[i]]][z],hash[sta[i][(z+1)%num[i]]][z+1],p[i]);
add(hash[n+1][z+1],T,inf);
while (tell()) while (tmp=zeng(S,inf)) ans+=tmp;
if (ans==k)
{
flag=1;
printf("%d\n",z+1);
break;
}
}
if (!flag) printf("%d\n",0);
}

浙公网安备 33010602011771号